### To Do
Copepod Collection
Copepods were collected at approximately weekly intervals from Lake
Champlain (Burlington Fishing Pier). Plankton was collected from the top
3 meters using a 250 um mesh net.
# Lake Champlain near Burlington, VT
siteNumber = "04294500"
ChamplainInfo = readNWISsite(siteNumber)
parameterCd = "00010"
startDate = "2023-01-01"
endDate = ""
#statCd = c("00001", "00002","00003", "00011") # 1 - max, 2 - min, 3 = mean
# Constructs the URL for the data wanted then downloads the data
url = constructNWISURL(siteNumbers = siteNumber, parameterCd = parameterCd,
startDate = startDate, endDate = endDate, service = "uv")
temp_data = importWaterML1(url, asDateTime = T) %>%
mutate("date" = as.Date(dateTime)) %>%
select(date, "temp" = X_00010_00000)
Collections began in late May 2023. Several gaps are present, but
collections have continued at roughly weekly intervals since then.
Copepods from 39 collections were used to make a total of 1086 thermal
limit measurements. Over this time period, collection temperatures
ranged from 2.5 to 26.5°C.
There is substantial variation in thermal limits across the species
collected. There is also some degree of variation within the species,
with thermal limits increasing slightly during the summer.
## Daily values for the period examined by dataset
collection_conditions = temp_data %>%
ungroup() %>%
group_by(date) %>%
summarise(mean_temp = mean(temp),
med_temp = median(temp),
var_temp = var(temp),
min_temp = min(temp),
max_temp = max(temp)) %>%
mutate("range_temp" = max_temp - min_temp,
date = as.Date(date)) %>%
ungroup() %>%
filter(date >= (min(as.Date(full_data$collection_date)) - 7))
## Mean female thermal limits for each species, grouped by collection
species_summaries = full_data %>%
#filter(sex == "female") %>%
group_by(sp_name, collection_date, collection_temp) %>%
summarise("mean_ctmax" = mean(ctmax),
"sample_size" = n(),
"ctmax_st_err" = (sd(ctmax) / sqrt(sample_size)),
"ctmax_var" = var(ctmax),
"mean_size" = mean(size),
"size_st_err" = (sd(size) / sqrt(sample_size)),
"size_var" = var(size)) %>%
ungroup() %>%
complete(sp_name, collection_date) %>%
arrange(desc(sample_size))
adult_summaries = full_data %>%
filter(sex == "female") %>%
group_by(sp_name, collection_date, collection_temp) %>%
summarise("mean_ctmax" = mean(ctmax),
"sample_size" = n(),
"ctmax_st_err" = (sd(ctmax) / sqrt(sample_size)),
"ctmax_var" = var(ctmax),
"mean_size" = mean(size),
"size_st_err" = (sd(size) / sqrt(sample_size)),
"size_var" = var(size)) %>%
ungroup() %>%
complete(sp_name, collection_date) %>%
arrange(desc(sample_size))
ggplot() +
geom_vline(data = unique(select(full_data, collection_date)),
aes(xintercept = as.Date(collection_date)),
colour = "grey90",
linewidth = 1) +
geom_line(data = collection_conditions,
aes(x = as.Date(date), y = mean_temp),
colour = "black",
linewidth = 2) +
# geom_errorbar(data = species_summaries,
# aes(x = as.Date(collection_date),
# ymin = mean_ctmax - ctmax_st_err, ymax = mean_ctmax + ctmax_st_err,
# colour = sp_name),
# position = position_dodge(width = 1),
# width = 5, linewidth = 1) +
# geom_point(data = adult_summaries,
# aes(x = as.Date(collection_date), y = mean_ctmax, colour = sp_name, size = sample_size)) +
geom_point(data = full_data,
aes(x = as.Date(collection_date), y = ctmax, colour = sp_name),
size = 2, position = position_jitter(width = 1, height = 0)) +
scale_colour_manual(values = species_cols) +
labs(x = "Date",
y = "Temperature (°C)",
colour = "Species",
size = "Sample Size") +
theme_matt() +
theme(legend.position = "right")

Size also varied, but primarily between rather than within
species.
ggplot() +
geom_vline(data = unique(select(full_data, collection_date)),
aes(xintercept = as.Date(collection_date)),
colour = "grey90",
linewidth = 1) +
geom_line(data = collection_conditions,
aes(x = as.Date(date), y = mean_temp),
colour = "black",
linewidth = 2) +
# geom_errorbar(data = species_summaries,
# aes(x = as.Date(collection_date),
# ymin = mean_ctmax - ctmax_st_err, ymax = mean_ctmax + ctmax_st_err,
# colour = sp_name),
# position = position_dodge(width = 1),
# width = 5, linewidth = 1) +
geom_point(data = adult_summaries,
aes(x = as.Date(collection_date), y = mean_size * 40, colour = sp_name, size = sample_size),
position = position_dodge(width = 1)) +
scale_colour_manual(values = species_cols) +
scale_y_continuous(
name = "Temperature", # Features of the first axis
sec.axis = sec_axis(~./40, name="Prosome Length (mm)"), # Add a second axis and specify its features
breaks = c(0,5,10,15,20,25,30)
) +
labs(x = "Date",
y = "Temperature (°C)",
colour = "Species") +
theme_matt() +
theme(legend.position = "right")

Shown below is CTmax and body size for the species with the most data
(Skistodiaptomus, L. minutus, L. sicilis, and
Epischura), plotted against the day of the year for each
sex/stage separately.
ctmax_feature = full_data %>%
mutate(doy = yday(collection_date)) %>%
filter(sp_name %in% c("Skistodiaptomus oregonensis", "Leptodiaptomus minutus", "Leptodiaptomus sicilis", "Epischura lacustris")) %>%
ggplot(aes(x = as.Date(collection_date), y = ctmax, colour = sp_name)) +
facet_grid(sp_name~sex) +
geom_point() +
scale_colour_manual(values = species_cols) +
labs(x = "Day of the Year",
y = "CTmax (°C)") +
theme_matt_facets() +
theme(axis.text.x = element_text(angle = 300, hjust = 0, vjust = 0.5),
legend.position = "none")
size_feature = full_data %>%
mutate(doy = yday(collection_date)) %>%
filter(sp_name %in% c("Skistodiaptomus oregonensis", "Leptodiaptomus minutus", "Leptodiaptomus sicilis", "Epischura lacustris")) %>%
ggplot(aes(x = as.Date(collection_date), y = size, colour = sp_name)) +
facet_grid(sp_name~sex) +
geom_point() +
scale_colour_manual(values = species_cols) +
labs(x = "Day of the Year",
y = "Size (mm)") +
theme_matt_facets() +
theme(axis.text.x = element_text(angle = 300, hjust = 0, vjust = 0.5),
legend.position = "none")
ggarrange(ctmax_feature, size_feature, common.legend = T, legend = "none")

adult_summaries %>%
ungroup() %>%
mutate(collection_num = as.numeric(factor(collection_date))) %>%
group_by(collection_date) %>%
arrange(collection_date) %>%
select(sp_name, collection_date, collection_num, sample_size) %>%
mutate(sample_size = replace_na(sample_size, 0)) %>%
mutate(total = sum(sample_size),
percentage = sample_size / total,
collection_date = lubridate::as_date(collection_date)) %>%
ggplot(aes(x = collection_date, y = percentage, fill = sp_name)) +
geom_area() +
scale_fill_manual(values = species_cols) +
scale_y_continuous(breaks = c(0,1)) +
labs(x = "Collection Date",
y = "Proportion",
fill = "Species") +
theme_minimal(base_size = 20) +
theme(panel.grid = element_blank(),
axis.ticks = element_line())

Temperature Variability
Lake Champlain is highly seasonal, with both average temperatures and
temperature variability changing throughout the year. These patterns in
the experienced thermal environment may drive the observed variation in
copepod thermal limits. However, the time period affecting copepod
thermal limits is unknown. Depending the on the duration of time
considered, there are large changes in the experienced environment, in
particular regarding the temperature range and variance. Consider for
example three time periods: the day of collection, one week prior to
collection, and four weeks prior to collection. While the overall
pattern is similar, we can see that, unsurprisingly, considering longer
periods of time results in larger ranges and slightly changes the
pattern of variance experienced.
## Daily values
daily_temp_data = temp_data %>%
ungroup() %>%
group_by(date) %>%
summarise(mean_temp = mean(temp),
med_temp = median(temp),
var_temp = var(temp),
min_temp = min(temp),
max_temp = max(temp)) %>%
mutate("range_temp" = max_temp - min_temp)
day_prior_temp_data = temp_data %>%
ungroup() %>%
group_by(date) %>%
summarise(mean_temp = mean(temp),
med_temp = median(temp),
var_temp = var(temp),
min_temp = min(temp),
max_temp = max(temp)) %>%
mutate(date = date + 1) %>%
rename_with(.fn = ~ paste0("prior_day_", .x), .cols = c(-date))
daily_plot = daily_temp_data %>%
pivot_longer(cols = c(-date),
names_to = "parameter",
values_to = "temp") %>%
ggplot(aes(x = date, y = temp, colour = parameter)) +
geom_line(linewidth = 1) +
scale_colour_manual(values = c(
"mean_temp" = "olivedrab3",
"med_temp" = "seagreen3",
"max_temp" = "tomato",
"min_temp" = "dodgerblue",
"range_temp" = "goldenrod3",
"var_temp" = "darkgoldenrod1"
)) +
scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
ggtitle("Daily Values") +
labs(y = "Temperature (°C)",
x = "") +
theme_bw(base_size = 20) +
theme(panel.grid = element_blank(),
axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
## Defining the function to get predictor values for periods of different lengths
get_predictors = function(daily_values, raw_temp, n_days){
prefix = str_replace_all(xfun::numbers_to_words(n_days), pattern = " ", replacement = "-")
mean_values = daily_values %>%
ungroup() %>%
mutate(mean_max = slide_vec(.x = max_temp, .f = mean, .before = n_days, .complete = T),
mean_min = slide_vec(.x = min_temp, .f = mean, .before = n_days, .complete = T),
mean_range = slide_vec(.x = range_temp, .f = mean, .before = n_days, .complete = T)) %>%
select(date, mean_max, mean_min, mean_range) %>%
rename_with( ~ paste(prefix, "day", .x, sep = "_"), .cols = c(-date))
period_values = raw_temp %>%
mutate(mean = slide_index_mean(temp, i = date, before = days(n_days),
na_rm = T),
max = slide_index_max(temp, i = date, before = days(n_days),
na_rm = T),
min = slide_index_min(temp, i = date, before = days(n_days),
na_rm = T),
med = slide_index_dbl(temp, .i = date, .before = days(n_days),
na_rm = T, .f = median),
var = slide_index_dbl(temp, .i = date, .before = days(n_days),
.f = var),
range = max - min) %>%
select(-temp) %>%
distinct() %>%
rename_with( ~ paste(prefix, "day", .x, sep = "_"), .cols = c(-date))%>%
inner_join(mean_values, by = c("date")) %>%
drop_na()
return(period_values)
}
# ## Getting predictor variables for different periods
#
# ### Short (three days)
# three_day_temps = get_predictors(daily_values = daily_temp_data,
# raw_temp = temp_data,
# n_days = 3)
#
# ### ONE WEEK
week_temps = get_predictors(daily_values = daily_temp_data,
raw_temp = temp_data,
n_days = 7)
week_plot = week_temps %>%
pivot_longer(cols = c(-date),
names_to = "parameter",
values_to = "temp") %>%
filter(parameter %in% c("seven_day_mean",
"seven_day_med",
"seven_day_max",
"seven_day_min",
"seven_day_var",
"seven_day_range")) %>%
mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
ggplot(aes(x = date, y = temp, colour = parameter)) +
geom_line(linewidth = 1) +
scale_colour_manual(values = c(
"mean_temp" = "olivedrab3",
"med_temp" = "seagreen3",
"max_temp" = "tomato",
"min_temp" = "dodgerblue",
"range_temp" = "goldenrod3",
"var_temp" = "darkgoldenrod1"
)) +
scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
ggtitle("One Week") +
labs(y = "Temperature (°C)",
x = "") +
theme_bw(base_size = 20) +
theme(panel.grid = element_blank(),
axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
#
# ### TWO WEEKS
# two_week_temps = get_predictors(daily_values = daily_temp_data,
# raw_temp = temp_data,
# n_days = 14)
#
# two_week_plot = two_week_temps %>%
# pivot_longer(cols = c(-date),
# names_to = "parameter",
# values_to = "temp") %>%
# filter(parameter %in% c("fourteen_day_mean",
# "fourteen_day_med",
# "fourteen_day_max",
# "fourteen_day_min",
# "fourteen_day_var",
# "fourteen_day_range")) %>%
# mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
# ggplot(aes(x = date, y = temp, colour = parameter)) +
# geom_line(linewidth = 1) +
# scale_colour_manual(values = c(
# "mean_temp" = "olivedrab3",
# "med_temp" = "seagreen3",
# "max_temp" = "tomato",
# "min_temp" = "dodgerblue",
# "range_temp" = "goldenrod3",
# "var_temp" = "darkgoldenrod1"
# )) +
# scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
# ggtitle("Two Weeks") +
# labs(y = "Temperature (°C)",
# x = "") +
# theme_bw(base_size = 20) +
# theme(panel.grid = element_blank(),
# axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
#
# ### FOUR WEEKS
four_week_temps = get_predictors(daily_values = daily_temp_data,
raw_temp = temp_data,
n_days = 28)
four_week_plot = four_week_temps %>%
pivot_longer(cols = c(-date),
names_to = "parameter",
values_to = "temp") %>%
filter(parameter %in% c("twenty-eight_day_mean",
"twenty-eight_day_med",
"twenty-eight_day_max",
"twenty-eight_day_min",
"twenty-eight_day_var",
"twenty-eight_day_range")) %>%
mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
ggplot(aes(x = date, y = temp, colour = parameter)) +
geom_line(linewidth = 1) +
scale_colour_manual(values = c(
"mean_temp" = "olivedrab3",
"med_temp" = "seagreen3",
"max_temp" = "tomato",
"min_temp" = "dodgerblue",
"range_temp" = "goldenrod3",
"var_temp" = "darkgoldenrod1"
)) +
scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
ggtitle("Four Weeks") +
labs(y = "Temperature (°C)",
x = "") +
theme_bw(base_size = 20) +
theme(panel.grid = element_blank(),
axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
#
# ### EIGHT WEEKS
# eight_week_temps = get_predictors(daily_values = daily_temp_data,
# raw_temp = temp_data,
# n_days = 56)
#
# eight_week_plot = eight_week_temps %>%
# pivot_longer(cols = c(-date),
# names_to = "parameter",
# values_to = "temp") %>%
# filter(parameter %in% c("fifty-six_day_mean",
# "fifty-six_day_med",
# "fifty-six_day_max",
# "fifty-six_day_min",
# "fifty-six_day_var",
# "fifty-six_day_range")) %>%
# mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
# ggplot(aes(x = date, y = temp, colour = parameter)) +
# geom_line(linewidth = 1) +
# scale_colour_manual(values = c(
# "mean_temp" = "olivedrab3",
# "med_temp" = "seagreen3",
# "max_temp" = "tomato",
# "min_temp" = "dodgerblue",
# "range_temp" = "goldenrod3",
# "var_temp" = "darkgoldenrod1"
# )) +
# scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
# ggtitle("Eight Weeks") +
# labs(y = "Temperature (°C)",
# x = "") +
# theme_bw(base_size = 20) +
# theme(panel.grid = element_blank(),
# axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
ggarrange(daily_plot, week_plot, four_week_plot,
common.legend = T, nrow = 1, legend = "bottom")

The different time periods examined by this climate data highlights
that the relationship between minimum and maximum temperatures changes
based on the window examined. For example, minimum and maximum
temperatures experienced over weekly intervals are closely linked,
whereas there is a distinct seasonal cycle in the relationship between
minimum and maximum temperatures experienced over periods of four
weeks.
one_week_doy_data = week_temps %>%
mutate(doy = yday(date))
one_week_temp_circle = ggplot(one_week_doy_data, aes(x = seven_day_mean_max, y = seven_day_mean_min, colour = doy)) +
geom_point() +
scale_colour_gradient2(
high = "dodgerblue4",
mid = "coral2",
low = "dodgerblue4",
midpoint = 182.5) +
labs(x = "Max. Temp. (°C)",
y = "Min. Temp. (°C)") +
labs(x = "Max. Temp. (°C)",
y = "Min. Temp. (°C)") +
ggtitle("One Week") +
theme_matt()
four_week_doy_data = four_week_temps %>%
mutate(doy = yday(date))
four_week_temp_circle = ggplot(four_week_doy_data, aes(x = `twenty-eight_day_max`, y = `twenty-eight_day_min`, colour = doy)) +
geom_point() +
scale_colour_gradient2(
high = "dodgerblue4",
mid = "coral2",
low = "dodgerblue4",
midpoint = 182.5) +
labs(x = "Max. Temp. (°C)",
y = "Min. Temp. (°C)") +
ggtitle("Four Week") +
theme_matt()
ggarrange(one_week_temp_circle, four_week_temp_circle,
common.legend = T, legend = "bottom")

The thermal environment over any period of time may drive patterns in
thermal acclimation. To explore the potential effects of different
acclimation windows, we examined the correlation between thermal limits
and different representations of the thermal environment for different
periods of time. Shown below are the correlation coefficients for these
relationships. Each facet shows the relationship for a different
dimension of the thermal environment. Correlation coefficients are
plotted for different durations, for species that were collected more
than five times. Only data for mature female copepods was included.
We can see that, in general, copepods are responding to proximate
cues from the thermal environment, with correlations generally dropping
off substantially as acclimation window duration increases. An exception
is Epischura lacustris, which appears to be responding to
maximum temperatures experienced over a 20 day time period.
### Pulling predictors and measuring correlations for much finer timescales; 1-56 days
num_colls = full_data %>%
filter(sex == "female") %>%
select(collection_date, sp_name) %>%
distinct() %>%
count(sp_name) %>%
filter(n >= 5)
corr_vals = data.frame()
dur_vals = c(1:50)
for(i in dur_vals){
duration_temps = get_predictors(daily_values = daily_temp_data,
raw_temp = temp_data,
n_days = i) %>%
filter(date %in% as_date(unique(full_data$collection_date)))
corr_data = full_data %>%
filter(sp_name %in% num_colls$sp_name) %>%
filter(sex == "female") %>%
mutate(collection_date = as.Date(collection_date)) %>%
inner_join(duration_temps, join_by(collection_date == date)) %>%
pivot_longer(cols = c(collection_temp, contains("day_")),
values_to = "value",
names_to = "predictor") %>%
group_by(sp_name, predictor) %>%
summarise(correlation = cor.test(ctmax, value)$estimate,
p.value = cor.test(ctmax, value)$p.value,
ci_low = cor.test(ctmax, value)$conf.int[1],
ci_high = cor.test(ctmax, value)$conf.int[2],
.groups = "keep") %>%
filter(predictor != "collection_temp") %>%
mutate(sig = ifelse(p.value <0.05, "Sig.", "Non Sig.")) %>%
separate(predictor, "_day_", into = c(NA, "parameter")) %>%
mutate(duration = i)
corr_vals = bind_rows(corr_vals, corr_data)
}
coll_corr = full_data %>%
filter(sp_name %in% num_colls$sp_name) %>%
filter(sex == "female") %>%
group_by(sp_name) %>%
summarise(correlation = cor.test(ctmax, collection_temp)$estimate,
p.value = cor.test(ctmax, collection_temp)$p.value,
ci_low = cor.test(ctmax, collection_temp)$conf.int[1],
ci_high = cor.test(ctmax, collection_temp)$conf.int[2]) %>%
mutate(sig = ifelse(p.value <0.05, "Sig.", "Non Sig.")) %>%
mutate(duration = 0,
parameter = "coll_temp")
corr_vals = corr_vals %>%
mutate(duration = as.numeric(duration)) %>%
bind_rows(coll_corr)
corr_vals %>%
mutate(parameter = fct_relevel(parameter, c("min", "max", "range",
"mean", "med", "var",
"mean_min", "mean_max", "mean_range"))) %>%
ggplot(aes(x = duration, y = correlation, colour = sp_name)) +
facet_wrap(.~parameter) +
geom_hline(yintercept = 0) +
geom_point(size = 0.9) +
geom_line(linewidth = 1.5) +
scale_colour_manual(values = species_cols) +
labs(x = "Duration (days)",
y = "Correlation",
colour = "Species") +
theme_matt_facets()

Shown here are the top three factors for each species.
#
# corr_vals = full_data %>%
# filter(sp_name %in% num_colls$sp_name) %>%
# filter(sex == "female") %>%
# mutate(collection_date = as.Date(collection_date)) %>%
# full_join(temp_predictors, join_by(collection_date == date)) %>%
# pivot_longer(cols = c(collection_temp, mean_temp:tail(names(.), 1)),
# values_to = "value",
# names_to = "predictor") %>%
# group_by(sp_name, predictor) %>%
# summarise(correlation = cor.test(ctmax, value)$estimate,
# p.value = cor.test(ctmax, value)$p.value,
# ci_low = cor.test(ctmax, value)$conf.int[1],
# ci_high = cor.test(ctmax, value)$conf.int[2]) %>%
# mutate(sig = ifelse(p.value <0.05, "Sig.", "Non Sig."))
corr_vals %>%
filter(sig == "Sig.") %>%
drop_na(correlation) %>%
group_by(sp_name) %>%
arrange(desc(correlation)) %>%
slice_head(n = 3) %>%
select("Species" = sp_name, "Predictor" = parameter, "Duration" = duration, "Correlation" = correlation, "P-Value" = p.value) %>%
knitr::kable(align = "c")
| Epischura lacustris |
max |
20 |
0.8926416 |
0.0000000 |
| Epischura lacustris |
max |
19 |
0.8906963 |
0.0000000 |
| Epischura lacustris |
max |
21 |
0.8874924 |
0.0000000 |
| Leptodiaptomus minutus |
max |
6 |
0.7547703 |
0.0000000 |
| Leptodiaptomus minutus |
max |
8 |
0.7547371 |
0.0000000 |
| Leptodiaptomus minutus |
max |
7 |
0.7545529 |
0.0000000 |
| Leptodiaptomus sicilis |
range |
23 |
0.3468229 |
0.0000000 |
| Leptodiaptomus sicilis |
range |
24 |
0.3457819 |
0.0000000 |
| Leptodiaptomus sicilis |
var |
21 |
0.3435719 |
0.0000000 |
| Limnocalanus macrurus |
max |
8 |
0.5542854 |
0.0001397 |
| Limnocalanus macrurus |
max |
7 |
0.5539836 |
0.0001412 |
| Limnocalanus macrurus |
mean_min |
6 |
0.5530991 |
0.0001454 |
| Senecella calanoides |
mean_max |
5 |
0.5434258 |
0.0445959 |
| Senecella calanoides |
max |
4 |
0.5402865 |
0.0460906 |
| Senecella calanoides |
max |
5 |
0.5402865 |
0.0460906 |
| Skistodiaptomus oregonensis |
max |
2 |
0.7964802 |
0.0000000 |
| Skistodiaptomus oregonensis |
max |
1 |
0.7903638 |
0.0000000 |
| Skistodiaptomus oregonensis |
mean_max |
2 |
0.7896567 |
0.0000000 |
Phenotypic variation (like acclimation of thermal limits) is a
physiological process. depending on the mechanistic underpinnings
(changes in HSP expression, etc.), the amount of time it takes for an
individual to acclimate may vary based on body size (larger species,
more cells, more time required to acclimate). Shown here is the duration
of the environmental acclimation window the copepods appear to be
responding to.
mean_sizes = full_data %>%
filter(sex == "female") %>%
group_by(sp_name) %>%
summarise(mean_size = mean(size, na.rm = T))
corr_vals %>%
group_by(sp_name) %>%
filter(correlation == max(correlation)) %>%
inner_join(mean_sizes, by = "sp_name") %>%
select(sp_name, duration, mean_size) %>%
ggplot(aes(x = mean_size, y = duration)) +
geom_point(aes(colour = sp_name),
size = 4) +
scale_colour_manual(values = species_cols) +
labs(x = "Mean Female Size (mm)",
y = "Acclimation Duration",
colour = "Species") +
theme_matt() +
theme(legend.position = "right")

Trait Variation
# ctmax_plot = full_data %>%
# mutate( #sp_name = str_replace(sp_name, pattern = " ",
# # replacement = "\n"),
# sp_name = fct_reorder(sp_name, ctmax, mean)) %>%
# ggplot(aes(y = sp_name, x = ctmax)) +
# geom_point(aes(colour= sp_name_sub),
# position = position_dodge(width = 0.3),
# size = 4) +
# scale_colour_manual(values = species_cols) +
# xlab(NULL) +
# labs(y = "",
# x = "CTmax (°C)",
# colour = "Group") +
# theme_matt() +
# theme(legend.position = "none")
#
# size_plot = full_data %>%
# mutate(sp_name = fct_reorder(sp_name, ctmax, mean)) %>%
# ggplot(aes(y = sp_name, x = size)) +
# geom_point(aes(colour= sp_name_sub),
# position = position_dodge(width = 0.3),
# size = 4) +
# scale_colour_manual(values = species_cols) +
# labs(x = "Prosome Length (mm)",
# y = "",
# colour = "Group") +
# guides(color = guide_legend(ncol = 1)) +
# theme_matt(base_size = ) +
# theme(legend.position = "right",
# axis.text.y = element_blank(),
# plot.margin = margin(0, 0, 0, 0,"cm"))
#
# trait_plot = ctmax_plot + size_plot
# trait_plot
Shown below are the clutch size distributions for the three
diaptomiid species, which produce egg sacs that allow for easy
quantification of fecundity.
full_data %>%
drop_na(fecundity) %>%
ggplot(aes(x = fecundity, fill = sp_name_sub)) +
facet_wrap(.~sp_name_sub, ncol = 1) +
geom_histogram(binwidth = 2) +
scale_fill_manual(values = species_cols) +
labs(x = "Fecundity (# Eggs)") +
theme_matt_facets() +
theme(legend.position = "none")

One of the main aims of this project is to examine the patterns and
processes driving variation in upper thermal limits across these species
of copepods.
Variation with temperature
We expect one of the primary drivers of copepod thermal limits to be
temperature. The correlation analysis has shown that the copepods are
generally (although not always) responding to the recent thermal
environment. Shown below are thermal limits, body size, and fecundity
values plotted against the temperature at the time of collection. Also
shown is warming tolerance, calculated as the difference between upper
thermal limit and the collection temperature.
We generally see an increase in thermal limits with increasing
collection temperature, a slight decrease in body size, and variable
relationships between temperature and fecundity. All species maintained
some degree of buffer between environmental temperatures and upper
thermal limits, but Epischura and L. minutus
approached their upper thermal limits during the warmest collections
during the summer.
ctmax_temp = ggplot(full_data, aes(x = collection_temp, y = ctmax, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "CTmax (°C)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
size_temp = ggplot(filter(full_data, sex != "juvenile"), aes(x = collection_temp, y = size, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "Length (mm)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
wt_temp = ggplot(full_data, aes(x = collection_temp, y = warming_tol, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "Warming Tolerance (°C)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
eggs_temp = ggplot(full_data, aes(x = collection_temp, y = fecundity, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "Fecundity (# Eggs)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
ggarrange(ctmax_temp, size_temp, wt_temp, eggs_temp,
common.legend = T, legend = "right")

full_data %>%
#filter(sex == "female") %>%
group_by(sp_name) %>% filter(n() > 5) %>% filter(!str_detect(sp_name, pattern = "kindti")) %>%
ggplot(aes(x = collection_temp, y = ctmax, colour = sp_name)) +
facet_wrap(sp_name~.) +
geom_point(size = 2, alpha = 0.8) +
geom_smooth(method = "lm", se = F, linewidth = 2) +
labs(x = "Collection Temp. (°C)",
y = "CTmax (°C)") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "none")

Temperature dependence is relatively weak in L. sicilis,
especially at cooler temperatures. We will return to this feature later
in the report, but for now we will note that there are two size morphs
in this species, which appear to respond differently to decreases in
temperature. There are significant differences between the morphs and
how temperature affects CTmax.
morph_data = full_data %>%
filter(sex == "female" & sp_name == "Leptodiaptomus sicilis") %>%
mutate(morph = if_else(size > 0.89, "large", "small"))
ggplot(morph_data, aes(x = collection_temp, y = ctmax, colour = morph)) +
geom_point(size = 2, alpha = 0.8) +
geom_smooth(method = "lm", se = T, linewidth = 2) +
labs(x = "Collection Temp. (°C)",
y = "CTmax (°C)") +
theme_matt() +
theme(legend.position = "none")

morph.model = lm(data = morph_data,
ctmax ~ collection_temp * morph)
knitr::kable(car::Anova(morph.model, type = "III", test = "F"))
| (Intercept) |
11346.62713 |
1 |
4238.107970 |
0.0000000 |
| collection_temp |
115.44002 |
1 |
43.118298 |
0.0000000 |
| morph |
36.75982 |
1 |
13.730254 |
0.0002546 |
| collection_temp:morph |
14.76000 |
1 |
5.513046 |
0.0195762 |
| Residuals |
744.28551 |
278 |
NA |
NA |
#summary(morph.model)
#morph.em = emmeans::emmeans(morph.model, pairwise ~ morph)
#plot(morph.em)
Copepods spent several days in lab during experiments. Shown below
are the CTmax residuals (taken from a model of CTmax against collection
temperature) plotted against the time spent in lab before measurements
were made. Individual regressions are shown for the residuals against
days in lab for each collection. We can see clearly that thermal limits
are fairly stable over time.
ggplot(ctmax_resids, aes(x = days_in_lab, y = resids, colour = sp_name, group = collection_date)) +
facet_wrap(sp_name~.) +
geom_point(size = 4, alpha = 0.5) +
geom_smooth(method = "lm", se = F, linewidth = 1) +
#scale_x_continuous(breaks = c(0:5)) +
labs(x = "Days in lab",
y = "CTmax Residuals") +
scale_colour_manual(values = species_cols) +
theme_matt_facets() +
theme(legend.position = "none")

full.model = lme4::lmer(data = model_data,
ctmax ~ sex + temp_cent + size_cent +
(1 + days_in_lab + temp_cent + size_cent|sp_name:sex))
car::Anova(full.model)
## Analysis of Deviance Table (Type II Wald chisquare tests)
##
## Response: ctmax
## Chisq Df Pr(>Chisq)
## sex 2.8465 2 0.2409
## temp_cent 42.1916 1 8.275e-11 ***
## size_cent 2.0467 1 0.1525
## ---
## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
fixed = fixef(full.model)
model_coefs = coefficients(full.model)$`sp_name:sex` %>%
rownames_to_column(var = "species") %>%
separate(species, into = c("species", "sex"), sep = ":") %>%
select(species, sex, "intercept" = "(Intercept)", temp_cent, size_cent, days_in_lab)
ggplot(model_coefs, aes(x = intercept, y = temp_cent)) +
geom_smooth(method = "lm", colour = "black") +
geom_point(aes(colour = species, shape = sex),
size = 6) +
scale_colour_manual(values = species_cols) +
labs(x = "Species Intercept",
y = "ARR") +
theme_matt() +
theme(legend.position = "right")

The term “acclimation response ratio” is often used to describe the
effect of temperature on thermal limits. The ARR is calculated as the
change in thermal limits per degree change in acclimation temperature.
For our data, we will estimate ARR as the slope of CTmax against
collection temperature. These slopes were taken from a regression of
CTmax against collection temperature and body size. Two different model
types were used, a simple linear regression and a mixed effects model.
The estimated ARR values were generally highly similar between the model
types used.
Sex and stage variation in thermal limits
Previous sections have generally lumped juvenile, female, and male
individuals together. There may be important stage- or sex-specific
differences in CTmax though. For several species, we have measurements
for individuals in different stages or of different sexes.
sex_sample_sizes = ctmax_resids %>%
group_by(sp_name, sex) %>%
summarise(num = n()) %>%
pivot_wider(id_cols = sp_name,
names_from = sex,
values_from = num,
values_fill = 0) %>%
select("Species" = sp_name, "Juvenile" = juvenile, "Female" = female, "Male" = male)
knitr::kable(sex_sample_sizes, align = "c")
| Epischura lacustris |
22 |
45 |
19 |
| Leptodiaptomus minutus |
10 |
209 |
33 |
| Leptodiaptomus sicilis |
31 |
282 |
78 |
| Limnocalanus macrurus |
2 |
42 |
36 |
| Osphranticum labronectum |
0 |
1 |
0 |
| Senecella calanoides |
1 |
14 |
8 |
| Skistodiaptomus oregonensis |
14 |
191 |
28 |
The female-male and female-juvenile comparisons show that there are
generally no differences in thermal limits between these groups.
ctmax_resids %>%
filter(sp_name %in% filter(sex_sample_sizes, Male > 0, Female > 0)$Species &
sex != "juvenile") %>%
ggplot(aes(x = sex, y = resids, colour = sp_name, group = sp_name)) +
facet_wrap(sp_name~., ncol = 2) +
geom_smooth(method = "lm", se = F, linewidth = 1) +
geom_point(size = 3,
alpha = 0.5,
position = position_jitter(height = 0, width = 0.05)) +
labs(x = "Sex",
y = "CTmax Residuals") +
scale_colour_manual(values = species_cols) +
theme_bw(base_size = 18) +
theme(legend.position = "none",
panel.grid = element_blank())

ctmax_resids %>%
filter(sp_name %in% filter(sex_sample_sizes, Juvenile > 0 & Female > 0)$Species &
sex != "male") %>%
ggplot(aes(x = sex, y = resids, colour = sp_name, group = sp_name)) +
facet_wrap(sp_name~., ncol = 2) +
geom_smooth(method = "lm", se = F, linewidth = 1) +
geom_point(size = 3,
alpha = 0.5,
position = position_jitter(height = 0, width = 0.05)) +
labs(x = "Sex",
y = "CTmax (°C)") +
scale_colour_manual(values = species_cols) +
theme_bw(base_size = 18) +
theme(legend.position = "none",
panel.grid = element_blank())

Trait Correlations and Trade-offs
A relationship between size and upper thermal limits has been
suggested in a wide range of other taxa. Shown below are the measured
upper thermal limits plotted against prosome length. The overall
relationship (inclusive of all species) is shown as the black line in
the background. Regressions for each individual species are also shown.
Across the entire assemblage, there is a strong decrease in thermal
limits with increasing size.
full_data %>%
#filter(sex == "female") %>%
ggplot( aes(x = size, y = ctmax, colour = sp_name)) +
geom_smooth(data = full_data,
aes(x = size, y = ctmax),
method = "lm",
colour ="black",
linewidth = 2.5) +
geom_point(size = 2, alpha = 0.3) +
geom_smooth(method = "lm", se = F, linewidth = 2) +
labs(x = "Length (mm)",
y = "CTmax (°C)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")

Shown here is the relationship for each species individually.
full_data %>%
#filter(sex == "female") %>%
group_by(sp_name) %>% filter(n() >2) %>% filter(!str_detect(sp_name, pattern = "kindti")) %>%
ggplot( aes(x = size, y = ctmax, colour = sp_name)) +
facet_wrap(sp_name~., scales = "free", nrow = 2) +
geom_point(size = 2, alpha = 0.8) +
geom_smooth(method = "lm", se = F, linewidth = 2) +
labs(x = "Length (mm)",
y = "CTmax (°C)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "none")

Shown below is the relationship between mean size and mean thermal
limits for females of each species. We see that larger species within
the community tend to have a lower thermal limit than smaller
species.
full_data %>%
group_by(sp_name, sex) %>%
summarize(mean_ctmax = mean(ctmax, na.rm = T),
mean_size = mean(size, na.rm = T)) %>%
#filter(sex == "female") %>%
ggplot(aes(x = mean_size, y = mean_ctmax)) +
geom_smooth(method = "lm", se = F, linewidth = 2, colour = "black") +
geom_point(aes(colour = sp_name, shape = sex),
size = 5) +
labs(x = "Length (mm)",
y = "CTmax (°C)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")

Shown here is the relationship between fecundity and size, showing
the classic pattern of increasing egg production with increasing
size.
ctmax_resids %>%
drop_na(fecundity) %>%
ggplot(aes(x = size, y = fecundity, colour = sp_name)) +
geom_smooth(method = "lm", se = F, linewidth = 2) +
geom_point(size = 2, alpha = 0.5) +
labs(x = "Prosome length (mm)",
y = "Fecundity (# Eggs)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")

Individuals may also allocate energy to different fitness related
traits, prioritizing reproductive output over environmental tolerance,
for example. Shown below is the relationship between CTmax residuals
(again, controlling for the effects of collection temperature) against
fecundity. We can see clearly that individuals with increased fecundity
are not decreasing thermal limits, suggesting that there is no energetic
trade-off between these traits.
ctmax_resids %>%
drop_na(fecundity) %>%
ggplot(aes(x = resids, y = fecundity, colour = sp_name)) +
geom_smooth(method = "lm", se = F, linewidth = 2) +
geom_point(size = 2, alpha = 0.5) +
labs(x = "CTmax Residuals",
y = "Fecundity (# Eggs)") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")

fitness.model = lm(data = ctmax_resids,
fecundity ~ resids * sp_name)
car::Anova(fitness.model)
## Anova Table (Type II tests)
##
## Response: fecundity
## Sum Sq Df F value Pr(>F)
## resids 4.5 1 0.2756 0.600038
## sp_name 8291.3 2 252.5993 < 2.2e-16 ***
## resids:sp_name 197.7 2 6.0234 0.002739 **
## Residuals 4693.8 286
## ---
## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
emmeans::emtrends(fitness.model, var = "resids","sp_name")
## sp_name resids.trend SE df lower.CL upper.CL
## Leptodiaptomus minutus 0.508 0.284 286 -0.0515 1.068
## Leptodiaptomus sicilis -0.129 0.250 286 -0.6217 0.364
## Skistodiaptomus oregonensis -1.213 0.407 286 -2.0147 -0.411
##
## Confidence level used: 0.95
Other patterns in variation
Leptodiaptomus sicilis is the most abundant species during
the winter. There was a large shift in the size of mature females
towards the end of December. These large and small individuals are the
same species (confirmed via COI sequencing), suggesting this shift may
reflect a transition from one generation to another and that, unlike in
many other lakes, there are two generations of L. sicilis per
year in Lake Champlain. This size difference may be caused by
differences in the developmental environments. For example, individuals
developing in January grow up at very low temperatures, and therefore
may reach larger sizes. These individuals oversummer in deep waters,
then re-emerge in October and produce a new generation. Water
temperatures are still fairly high through November, which results in a
generation of smaller individuals, which mature in time to produce a new
generation in January.
Shown below is the distribution of pairwise distances between COI
sequences of large and small morphs. Distances in both within- and
across-morph comparisons are small.
ind_dist = ape::dist.dna(sic_dnabin, model = "raw") %>% as.matrix %>%
as_tibble() %>%
mutate("ind1" = colnames(.)) %>%
pivot_longer(-ind1, names_to = "ind2", values_to = "dist") %>%
mutate(ind1 = factor(ind1),
ind2 = factor(ind2)) %>%
filter(!(ind1 == "sore1" | ind2 == "sore1")) %>%
mutate(
ind1 = case_when(
ind1 == "S1" ~ "small1",
ind1 == "S3" ~ "small3",
ind1 == "lsic3" ~ "small4",
ind1 == "lsic5" ~ "small6",
ind1 == "lsic9" ~ "small8",
ind1 == "lsic10" ~ "small9",
ind1 == "lsic11" ~ "small10",
ind1 == "L1" ~ "large1",
ind1 == "L2" ~ "large2",
ind1 == "L3" ~ "large3",
ind1 == "lsic1" ~ "large4",
ind1 == "lsic2" ~ "large5",
ind1 == "lsic7" ~ "large6",
ind1 == "lsic8" ~ "large7"),
ind2 = case_when(
ind2 == "S1" ~ "small1",
ind2 == "S3" ~ "small3",
ind2 == "lsic3" ~ "small4",
ind2 == "lsic5" ~ "small6",
ind2 == "lsic9" ~ "small8",
ind2 == "lsic10" ~ "small9",
ind2 == "lsic11" ~ "small10",
ind2 == "L1" ~ "large1",
ind2 == "L2" ~ "large2",
ind2 == "L3" ~ "large3",
ind2 == "lsic1" ~ "large4",
ind2 == "lsic2" ~ "large5",
ind2 == "lsic7" ~ "large6",
ind2 == "lsic8" ~ "large7"),
'comparison' = case_when(
str_detect(ind1, pattern = "large") & str_detect(ind2, pattern = "large") ~ "within",
str_detect(ind1, pattern = "small") & str_detect(ind2, pattern = "small") ~ "within",
str_detect(ind1, pattern = "large") & str_detect(ind2, pattern = "small") ~ "across",
str_detect(ind1, pattern = "small") & str_detect(ind2, pattern = "large") ~ "across"
))
ggplot(ind_dist, aes(dist, fill = comparison)) +
geom_histogram(binwidth = 0.005) +
labs(x = "Distance") +
theme_matt()

full_data %>%
filter(sp_name == "Leptodiaptomus sicilis") %>%
filter(sex == "female") %>%
group_by(collection_date) %>%
mutate(size_center = scale(size, center = T, scale = F)) %>%
ggplot(aes(y = collection_date, x = size, fill = collection_temp)) +
geom_density_ridges(bandwidth = 0.04) +
geom_vline(xintercept = 0.89) +
labs(x = "Size (mm)",
y = "Date",
fill = "Coll. Temp. (°C)") +
theme_matt() +
theme(legend.position = "right",
axis.text.y = element_text(size = 12))

full_data %>%
filter(sp_name == "Leptodiaptomus minutus") %>%
filter(sex == "female") %>%
group_by(collection_date) %>%
mutate(size_center = scale(size, center = T, scale = F)) %>%
ggplot(aes(y = collection_date, x = size, fill = collection_temp)) +
geom_density_ridges(bandwidth = 0.04) +
geom_vline(xintercept = 0.69) +
labs(x = "Size (mm)",
y = "Date",
fill = "Coll. Temp. (°C)") +
coord_cartesian(xlim = c(0.5,0.9)) +
theme_matt() +
theme(legend.position = "right",
axis.text.y = element_text(size = 12))

if(predict_vuln == F){
knitr::knit_exit()
}
LS0tCnRpdGxlOiBTZWFzb25hbGl0eSBpbiBMYWtlIENoYW1wbGFpbiBDb3BlcG9kIFRoZXJtYWwgTGltaXRzCmRhdGU6ICJgciBTeXMuRGF0ZSgpYCIKb3V0cHV0OiAKICBodG1sX2RvY3VtZW50OgogICAgICAgICAgY29kZV9mb2xkaW5nOiBoaWRlCiAgICAgICAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCiAgICAgICAgICB0b2M6IHRydWUKICAgICAgICAgIHRvY19mbG9hdDogdHJ1ZQogIGdpdGh1Yl9kb2N1bWVudDoKICAgICAgICAgIGh0bWxfcHJldmlldzogZmFsc2UKICAgICAgICAgIHRvYzogdHJ1ZQogICAgICAgICAgdG9jX2RlcHRoOiAzCi0tLQoKYGBge3IgdG8tZG99CiMjIyBUbyBEbyAKCmBgYAoKCmBgYHtyIHNldHVwLCBpbmNsdWRlPVQsIG1lc3NhZ2UgPSBGLCB3YXJuaW5nID0gRiwgZWNobyA9IEZ9CmtuaXRyOjpvcHRzX2NodW5rJHNldCgKICBlY2hvID0ga25pdHI6OmlzX2h0bWxfb3V0cHV0KCksCiAgZmlnLmFsaWduID0gImNlbnRlciIsCiAgZmlnLnBhdGggPSAiLi4vRmlndXJlcy9tYXJrZG93bi8iLAogIGRldiA9IGMoInBuZyIsICJwZGYiKSwKICBtZXNzYWdlID0gRkFMU0UsCiAgd2FybmluZyA9IEZBTFNFLAogIGNvbGxhcHNlID0gVAopCgp0aGVtZV9tYXR0ID0gZnVuY3Rpb24oYmFzZV9zaXplID0gMTgsCiAgICAgICAgICAgICAgICAgICAgICBkYXJrX3RleHQgPSAiZ3JleTIwIil7CiAgbWlkX3RleHQgPC0gIG1vbm9jaHJvbWVSOjpnZW5lcmF0ZV9wYWxldHRlKGRhcmtfdGV4dCwgImdvX2xpZ2h0ZXIiLCBuX2NvbG91cnMgPSA1KVsyXQogIGxpZ2h0X3RleHQgPC0gIG1vbm9jaHJvbWVSOjpnZW5lcmF0ZV9wYWxldHRlKGRhcmtfdGV4dCwgImdvX2xpZ2h0ZXIiLCBuX2NvbG91cnMgPSA1KVszXQogIAogIGdncHVicjo6dGhlbWVfcHVicihiYXNlX2ZhbWlseT0ic2FucyIpICUrcmVwbGFjZSUgCiAgICB0aGVtZSgKICAgICAgcGFuZWwuYmFja2dyb3VuZCAgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLCAKICAgICAgcGxvdC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGw9InRyYW5zcGFyZW50IiwgY29sb3VyPU5BKSwgCiAgICAgIGxlZ2VuZC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGw9InRyYW5zcGFyZW50IiwgY29sb3VyPU5BKSwKICAgICAgbGVnZW5kLmtleSA9IGVsZW1lbnRfcmVjdChmaWxsPSJ0cmFuc3BhcmVudCIsIGNvbG91cj1OQSksCiAgICAgIHRleHQgPSBlbGVtZW50X3RleHQoY29sb3VyID0gbWlkX3RleHQsIGxpbmVoZWlnaHQgPSAxLjEpLAogICAgICB0aXRsZSA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMS41LAogICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBkYXJrX3RleHQpLAogICAgICBheGlzLnRleHQgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG91ciA9IG1pZF90ZXh0KSwKICAgICAgYXhpcy50aXRsZS54ID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAxLjIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtYXJnaW4gPSB1bml0KGMoMywgMCwgMCwgMCksICJtbSIpKSwKICAgICAgYXhpcy50aXRsZS55ID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAxLjIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtYXJnaW4gPSB1bml0KGMoMCwgNSwgMCwgMCksICJtbSIpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFuZ2xlID0gOTApLAogICAgICBsZWdlbmQudGV4dCA9IGVsZW1lbnRfdGV4dChzaXplPWJhc2Vfc2l6ZSAqIDAuOSksCiAgICAgIGxlZ2VuZC50aXRsZSA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMC45LCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZhY2UgPSAiYm9sZCIpLAogICAgICBwbG90Lm1hcmdpbiA9IG1hcmdpbigwLjI1LCAwLjI1LCAwLjI1LCAwLjI1LCJjbSIpCiAgICApCn0KCnRoZW1lX21hdHRfZmFjZXRzID0gZnVuY3Rpb24oYmFzZV9zaXplID0gMTgsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGFya190ZXh0ID0gImdyZXkyMCIpewogIG1pZF90ZXh0IDwtICBtb25vY2hyb21lUjo6Z2VuZXJhdGVfcGFsZXR0ZShkYXJrX3RleHQsICJnb19saWdodGVyIiwgbl9jb2xvdXJzID0gNSlbMl0KICBsaWdodF90ZXh0IDwtICBtb25vY2hyb21lUjo6Z2VuZXJhdGVfcGFsZXR0ZShkYXJrX3RleHQsICJnb19saWdodGVyIiwgbl9jb2xvdXJzID0gNSlbM10KICAKICB0aGVtZV9idyhiYXNlX2ZhbWlseT0ic2FucyIpICUrcmVwbGFjZSUgCiAgICB0aGVtZSgKICAgICAgcGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSwKICAgICAgcGFuZWwuYmFja2dyb3VuZCAgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLCAKICAgICAgcGxvdC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGw9InRyYW5zcGFyZW50IiwgY29sb3VyPU5BKSwgCiAgICAgIGxlZ2VuZC5iYWNrZ3JvdW5kID0gZWxlbWVudF9yZWN0KGZpbGw9InRyYW5zcGFyZW50IiwgY29sb3VyPU5BKSwKICAgICAgbGVnZW5kLmtleSA9IGVsZW1lbnRfcmVjdChmaWxsPSJ0cmFuc3BhcmVudCIsIGNvbG91cj1OQSksCiAgICAgIHRleHQgPSBlbGVtZW50X3RleHQoY29sb3VyID0gbWlkX3RleHQsIGxpbmVoZWlnaHQgPSAxLjEpLAogICAgICBzdHJpcC50ZXh0LnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSksCiAgICAgIHRpdGxlID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAxLjUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG91ciA9IGRhcmtfdGV4dCksCiAgICAgIGF4aXMudGV4dCA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sb3VyID0gbWlkX3RleHQpLAogICAgICBheGlzLnRpdGxlLnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDEuMiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1hcmdpbiA9IHVuaXQoYygzLCAwLCAwLCAwKSwgIm1tIikpLAogICAgICBheGlzLnRpdGxlLnkgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDEuMiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1hcmdpbiA9IHVuaXQoYygwLCA1LCAwLCAwKSwgIm1tIiksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYW5nbGUgPSA5MCksCiAgICAgIGxlZ2VuZC50ZXh0ID0gZWxlbWVudF90ZXh0KHNpemU9YmFzZV9zaXplICogMC45KSwKICAgICAgbGVnZW5kLnRpdGxlID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAwLjksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmFjZSA9ICJib2xkIiksCiAgICAgIHBsb3QubWFyZ2luID0gbWFyZ2luKDAuMjUsIDAuMjUsIDAuMjUsIDAuMjUsImNtIikKICAgICkKfQoKc3BlY2llc19jb2xzID0gYygiTGVwdG9kaWFwdG9tdXMgbWludXR1cyIgPSAiI2ZmZDAyOSIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMganV2ZW5pbGUiID0gIiNlM2Q4YWYiLAogICAgICAgICAgICAgICAgICJMZXB0b2RpYXB0b211cyBtaW51dHVzIG1hbGUiID0gIiNmZmU4OTYiLAogICAgICAgICAgICAgICAgICJMZXB0b2RpYXB0b211cyBzaWNpbGlzIiA9ICIjRDg2RjI5IiwKICAgICAgICAgICAgICAgICAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyBtYWxlIiA9ICIjRTI4QzAwIiwKICAgICAgICAgICAgICAgICAiU2tpc3RvZGlhcHRvbXVzIG9yZWdvbmVuc2lzIiA9ICIjQzVDMzVBIiwKICAgICAgICAgICAgICAgICAiU2tpc3RvZGlhcHRvbXVzIG9yZWdvbmVuc2lzIG1hbGUiID0gIiNlNmU2YWEiLCAKICAgICAgICAgICAgICAgICAiRXBpc2NodXJhIGxhY3VzdHJpcyBqdXZlbmlsZSIgPSAicGx1bTEiLCAKICAgICAgICAgICAgICAgICAiRXBpc2NodXJhIGxhY3VzdHJpcyBtYWxlIiA9ICJwbHVtMyIsIAogICAgICAgICAgICAgICAgICJFcGlzY2h1cmEgbGFjdXN0cmlzIiA9ICJwbHVtNCIsIAogICAgICAgICAgICAgICAgICJMaW1ub2NhbGFudXMgbWFjcnVydXMiID0gInNreWJsdWU0IiwgCiAgICAgICAgICAgICAgICAgIkxpbW5vY2FsYW51cyBtYWNydXJ1cyBtYWxlIiA9ICJza3libHVlMyIsIAogICAgICAgICAgICAgICAgICJMaW1ub2NhbGFudXMgbWFjcnVydXMganV2ZW5pbGUiID0gInNreWJsdWUiLCAKICAgICAgICAgICAgICAgICAiU2VuZWNlbGxhIGNhbGFub2lkZXMiID0gImRhcmtzZWFncmVlbjMiLAogICAgICAgICAgICAgICAgICJMZXB0b2RvcmEga2luZHRpIG1hbGUiID0gImxpZ2h0Ymx1ZTMiLAogICAgICAgICAgICAgICAgICJMZXB0b2RvcmEga2luZHRpIiA9ICJsaWdodGJsdWU0IiwKICAgICAgICAgICAgICAgICAiTGVwdG9kb3JhIGtpbmR0aSBqdXZlbmlsZSIgPSAibGlnaHRibHVlIiwKICAgICAgICAgICAgICAgICAiT3NwaHJhbnRpY3VtIGxhYnJvbmVjdHVtIiA9ICJsaWdodGNvcmFsIikKYGBgCgojIyBDb3BlcG9kIENvbGxlY3Rpb24KCkNvcGVwb2RzIHdlcmUgY29sbGVjdGVkIGF0IGFwcHJveGltYXRlbHkgd2Vla2x5IGludGVydmFscyBmcm9tIExha2UgQ2hhbXBsYWluIChCdXJsaW5ndG9uIEZpc2hpbmcgUGllcikuIFBsYW5rdG9uIHdhcyBjb2xsZWN0ZWQgZnJvbSB0aGUgdG9wIDMgbWV0ZXJzIHVzaW5nIGEgMjUwIHVtIG1lc2ggbmV0LiAKCmBgYHtyfQojIExha2UgQ2hhbXBsYWluIG5lYXIgQnVybGluZ3RvbiwgVlQKc2l0ZU51bWJlciA9ICIwNDI5NDUwMCIKQ2hhbXBsYWluSW5mbyA9IHJlYWROV0lTc2l0ZShzaXRlTnVtYmVyKQpwYXJhbWV0ZXJDZCA9ICIwMDAxMCIKc3RhcnREYXRlID0gIjIwMjMtMDEtMDEiCmVuZERhdGUgPSAiIgojc3RhdENkID0gYygiMDAwMDEiLCAiMDAwMDIiLCIwMDAwMyIsICIwMDAxMSIpICMgMSAtIG1heCwgMiAtIG1pbiwgMyA9IG1lYW4KCiMgQ29uc3RydWN0cyB0aGUgVVJMIGZvciB0aGUgZGF0YSB3YW50ZWQgdGhlbiBkb3dubG9hZHMgdGhlIGRhdGEKdXJsID0gY29uc3RydWN0TldJU1VSTChzaXRlTnVtYmVycyA9IHNpdGVOdW1iZXIsIHBhcmFtZXRlckNkID0gcGFyYW1ldGVyQ2QsIAogICAgICAgICAgICAgICAgICAgICAgIHN0YXJ0RGF0ZSA9IHN0YXJ0RGF0ZSwgZW5kRGF0ZSA9IGVuZERhdGUsIHNlcnZpY2UgPSAidXYiKQoKdGVtcF9kYXRhID0gaW1wb3J0V2F0ZXJNTDEodXJsLCBhc0RhdGVUaW1lID0gVCkgJT4lIAogIG11dGF0ZSgiZGF0ZSIgPSBhcy5EYXRlKGRhdGVUaW1lKSkgJT4lIAogIHNlbGVjdChkYXRlLCAidGVtcCIgPSBYXzAwMDEwXzAwMDAwKQpgYGAKCkNvbGxlY3Rpb25zIGJlZ2FuIGluIGxhdGUgTWF5IDIwMjMuIFNldmVyYWwgZ2FwcyBhcmUgcHJlc2VudCwgYnV0IGNvbGxlY3Rpb25zIGhhdmUgY29udGludWVkIGF0IHJvdWdobHkgd2Vla2x5IGludGVydmFscyBzaW5jZSB0aGVuLiBDb3BlcG9kcyBmcm9tIGByIGxlbmd0aCh1bmlxdWUoZnVsbF9kYXRhJGNvbGxlY3Rpb25fZGF0ZSkpYCBjb2xsZWN0aW9ucyB3ZXJlIHVzZWQgdG8gbWFrZSBhIHRvdGFsIG9mIGByIGRpbShmdWxsX2RhdGEpWzFdYCB0aGVybWFsIGxpbWl0IG1lYXN1cmVtZW50cy4gT3ZlciB0aGlzIHRpbWUgcGVyaW9kLCBjb2xsZWN0aW9uIHRlbXBlcmF0dXJlcyByYW5nZWQgZnJvbSBgciBwYXN0ZShtaW4oZnVsbF9kYXRhJGNvbGxlY3Rpb25fdGVtcCksICIgdG8gIiwgbWF4KGZ1bGxfZGF0YSRjb2xsZWN0aW9uX3RlbXApLCBzZXAgPSAiIilgwrBDLiAgICAgCgpUaGVyZSBpcyBzdWJzdGFudGlhbCB2YXJpYXRpb24gaW4gdGhlcm1hbCBsaW1pdHMgYWNyb3NzIHRoZSBzcGVjaWVzIGNvbGxlY3RlZC4gVGhlcmUgaXMgYWxzbyBzb21lIGRlZ3JlZSBvZiB2YXJpYXRpb24gd2l0aGluIHRoZSBzcGVjaWVzLCB3aXRoIHRoZXJtYWwgbGltaXRzIGluY3JlYXNpbmcgc2xpZ2h0bHkgZHVyaW5nIHRoZSBzdW1tZXIuICAgIAoKYGBge3IgY3RtYXgtdGltZXNlcmllcywgZmlnLndpZHRoPTEwLCBmaWcuaGVpZ2h0PTV9CiMjIERhaWx5IHZhbHVlcyBmb3IgdGhlIHBlcmlvZCBleGFtaW5lZCBieSBkYXRhc2V0CmNvbGxlY3Rpb25fY29uZGl0aW9ucyA9IHRlbXBfZGF0YSAlPiUKICB1bmdyb3VwKCkgJT4lIAogIGdyb3VwX2J5KGRhdGUpICU+JSAKICBzdW1tYXJpc2UobWVhbl90ZW1wID0gbWVhbih0ZW1wKSwKICAgICAgICAgICAgbWVkX3RlbXAgPSBtZWRpYW4odGVtcCksCiAgICAgICAgICAgIHZhcl90ZW1wID0gdmFyKHRlbXApLCAKICAgICAgICAgICAgbWluX3RlbXAgPSBtaW4odGVtcCksIAogICAgICAgICAgICBtYXhfdGVtcCA9IG1heCh0ZW1wKSkgJT4lIAogIG11dGF0ZSgicmFuZ2VfdGVtcCIgPSBtYXhfdGVtcCAtIG1pbl90ZW1wLAogICAgICAgICBkYXRlID0gYXMuRGF0ZShkYXRlKSkgJT4lIAogIHVuZ3JvdXAoKSAlPiUgIAogIGZpbHRlcihkYXRlID49IChtaW4oYXMuRGF0ZShmdWxsX2RhdGEkY29sbGVjdGlvbl9kYXRlKSkgLSA3KSkKCiMjIE1lYW4gZmVtYWxlIHRoZXJtYWwgbGltaXRzIGZvciBlYWNoIHNwZWNpZXMsIGdyb3VwZWQgYnkgY29sbGVjdGlvbgpzcGVjaWVzX3N1bW1hcmllcyA9IGZ1bGxfZGF0YSAlPiUgIAogICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXApICU+JSAgCiAgc3VtbWFyaXNlKCJtZWFuX2N0bWF4IiA9IG1lYW4oY3RtYXgpLAogICAgICAgICAgICAic2FtcGxlX3NpemUiID0gbigpLAogICAgICAgICAgICAiY3RtYXhfc3RfZXJyIiA9IChzZChjdG1heCkgLyBzcXJ0KHNhbXBsZV9zaXplKSksCiAgICAgICAgICAgICJjdG1heF92YXIiID0gdmFyKGN0bWF4KSwgCiAgICAgICAgICAgICJtZWFuX3NpemUiID0gbWVhbihzaXplKSwKICAgICAgICAgICAgInNpemVfc3RfZXJyIiA9IChzZChzaXplKSAvIHNxcnQoc2FtcGxlX3NpemUpKSwKICAgICAgICAgICAgInNpemVfdmFyIiA9IHZhcihzaXplKSkgJT4lICAKICB1bmdyb3VwKCkgJT4lIAogIGNvbXBsZXRlKHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSkgJT4lIAogIGFycmFuZ2UoZGVzYyhzYW1wbGVfc2l6ZSkpCgphZHVsdF9zdW1tYXJpZXMgPSBmdWxsX2RhdGEgJT4lICAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXApICU+JSAgCiAgc3VtbWFyaXNlKCJtZWFuX2N0bWF4IiA9IG1lYW4oY3RtYXgpLAogICAgICAgICAgICAic2FtcGxlX3NpemUiID0gbigpLAogICAgICAgICAgICAiY3RtYXhfc3RfZXJyIiA9IChzZChjdG1heCkgLyBzcXJ0KHNhbXBsZV9zaXplKSksCiAgICAgICAgICAgICJjdG1heF92YXIiID0gdmFyKGN0bWF4KSwgCiAgICAgICAgICAgICJtZWFuX3NpemUiID0gbWVhbihzaXplKSwKICAgICAgICAgICAgInNpemVfc3RfZXJyIiA9IChzZChzaXplKSAvIHNxcnQoc2FtcGxlX3NpemUpKSwKICAgICAgICAgICAgInNpemVfdmFyIiA9IHZhcihzaXplKSkgJT4lICAKICB1bmdyb3VwKCkgJT4lIAogIGNvbXBsZXRlKHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSkgJT4lIAogIGFycmFuZ2UoZGVzYyhzYW1wbGVfc2l6ZSkpCgoKZ2dwbG90KCkgKyAKICBnZW9tX3ZsaW5lKGRhdGEgPSB1bmlxdWUoc2VsZWN0KGZ1bGxfZGF0YSwgY29sbGVjdGlvbl9kYXRlKSksIAogICAgICAgICAgICAgYWVzKHhpbnRlcmNlcHQgPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSkpLAogICAgICAgICAgICAgY29sb3VyID0gImdyZXk5MCIsCiAgICAgICAgICAgICBsaW5ld2lkdGggPSAxKSArIAogIGdlb21fbGluZShkYXRhID0gY29sbGVjdGlvbl9jb25kaXRpb25zLCAKICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGRhdGUpLCB5ID0gbWVhbl90ZW1wKSwKICAgICAgICAgICAgY29sb3VyID0gImJsYWNrIiwgCiAgICAgICAgICAgIGxpbmV3aWR0aCA9IDIpICsgCiAgIyBnZW9tX2Vycm9yYmFyKGRhdGEgPSBzcGVjaWVzX3N1bW1hcmllcywKICAjICAgICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksCiAgIyAgICAgICAgICAgICAgICAgICB5bWluID0gbWVhbl9jdG1heCAtIGN0bWF4X3N0X2VyciwgeW1heCA9IG1lYW5fY3RtYXggKyBjdG1heF9zdF9lcnIsCiAgIyAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBzcF9uYW1lKSwKICAjICAgICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDEpLAogICMgICAgICAgICAgICAgICB3aWR0aCA9IDUsIGxpbmV3aWR0aCA9IDEpICsKICAjIGdlb21fcG9pbnQoZGF0YSA9IGFkdWx0X3N1bW1hcmllcywgCiAgIyAgICAgICAgICAgIGFlcyh4ID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpLCB5ID0gbWVhbl9jdG1heCwgY29sb3VyID0gc3BfbmFtZSwgc2l6ZSA9IHNhbXBsZV9zaXplKSkgKyAKICBnZW9tX3BvaW50KGRhdGEgPSBmdWxsX2RhdGEsIAogICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBjdG1heCwgY29sb3VyID0gc3BfbmFtZSksCiAgICAgICAgICAgICBzaXplID0gMiwgcG9zaXRpb24gPSBwb3NpdGlvbl9qaXR0ZXIod2lkdGggPSAxLCBoZWlnaHQgPSAwKSkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiRGF0ZSIsIAogICAgICAgeSA9ICJUZW1wZXJhdHVyZSAowrBDKSIsIAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiLAogICAgICAgc2l6ZSA9ICJTYW1wbGUgU2l6ZSIpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgpTaXplIGFsc28gdmFyaWVkLCBidXQgcHJpbWFyaWx5IGJldHdlZW4gcmF0aGVyIHRoYW4gd2l0aGluIHNwZWNpZXMuIAoKYGBge3Igc2l6ZS10aW1lc2VyaWVzLCBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9NX0KZ2dwbG90KCkgKyAKICBnZW9tX3ZsaW5lKGRhdGEgPSB1bmlxdWUoc2VsZWN0KGZ1bGxfZGF0YSwgY29sbGVjdGlvbl9kYXRlKSksIAogICAgICAgICAgICAgYWVzKHhpbnRlcmNlcHQgPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSkpLAogICAgICAgICAgICAgY29sb3VyID0gImdyZXk5MCIsCiAgICAgICAgICAgICBsaW5ld2lkdGggPSAxKSArIAogIGdlb21fbGluZShkYXRhID0gY29sbGVjdGlvbl9jb25kaXRpb25zLCAKICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGRhdGUpLCB5ID0gbWVhbl90ZW1wKSwKICAgICAgICAgICAgY29sb3VyID0gImJsYWNrIiwgCiAgICAgICAgICAgIGxpbmV3aWR0aCA9IDIpICsgCiAgIyBnZW9tX2Vycm9yYmFyKGRhdGEgPSBzcGVjaWVzX3N1bW1hcmllcywKICAjICAgICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIAogICMgICAgICAgICAgICAgICAgICAgeW1pbiA9IG1lYW5fY3RtYXggLSBjdG1heF9zdF9lcnIsIHltYXggPSBtZWFuX2N0bWF4ICsgY3RtYXhfc3RfZXJyLAogICMgICAgICAgICAgICAgICAgICAgY29sb3VyID0gc3BfbmFtZSksCiAgIyAgICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fZG9kZ2Uod2lkdGggPSAxKSwKICAjICAgICAgICAgICAgICAgd2lkdGggPSA1LCBsaW5ld2lkdGggPSAxKSArIAogIGdlb21fcG9pbnQoZGF0YSA9IGFkdWx0X3N1bW1hcmllcywgCiAgICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwgeSA9IG1lYW5fc2l6ZSAqIDQwLCBjb2xvdXIgPSBzcF9uYW1lLCBzaXplID0gc2FtcGxlX3NpemUpLAogICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDEpKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHNjYWxlX3lfY29udGludW91cygKICAgIG5hbWUgPSAiVGVtcGVyYXR1cmUiLCAjIEZlYXR1cmVzIG9mIHRoZSBmaXJzdCBheGlzCiAgICBzZWMuYXhpcyA9IHNlY19heGlzKH4uLzQwLCBuYW1lPSJQcm9zb21lIExlbmd0aCAobW0pIiksICMgQWRkIGEgc2Vjb25kIGF4aXMgYW5kIHNwZWNpZnkgaXRzIGZlYXR1cmVzCiAgICBicmVha3MgPSBjKDAsNSwxMCwxNSwyMCwyNSwzMCkKICApICsgCiAgbGFicyh4ID0gIkRhdGUiLCAKICAgICAgIHkgPSAiVGVtcGVyYXR1cmUgKMKwQykiLCAKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQpgYGAKClNob3duIGJlbG93IGlzIENUbWF4IGFuZCBib2R5IHNpemUgZm9yIHRoZSBzcGVjaWVzIHdpdGggdGhlIG1vc3QgZGF0YSAoKlNraXN0b2RpYXB0b211cyosICpMLiBtaW51dHVzKiwgKkwuIHNpY2lsaXMqLCBhbmQgKkVwaXNjaHVyYSopLCBwbG90dGVkIGFnYWluc3QgdGhlIGRheSBvZiB0aGUgeWVhciBmb3IgZWFjaCBzZXgvc3RhZ2Ugc2VwYXJhdGVseS4gCgpgYGB7ciB0cmFpdC1kb3ktZmVhdHVyZSwgZmlnLndpZHRoID0gMTQsIGZpZy5oZWlnaHQgPSA3fQpjdG1heF9mZWF0dXJlID0gZnVsbF9kYXRhICU+JSAgCiAgbXV0YXRlKGRveSA9IHlkYXkoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogIGZpbHRlcihzcF9uYW1lICVpbiUgYygiU2tpc3RvZGlhcHRvbXVzIG9yZWdvbmVuc2lzIiwgIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMiLCAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyIsICJFcGlzY2h1cmEgbGFjdXN0cmlzIikpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBjdG1heCwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfZ3JpZChzcF9uYW1lfnNleCkgKyAKICBnZW9tX3BvaW50KCkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiRGF5IG9mIHRoZSBZZWFyIiwgCiAgICAgICB5ID0gIkNUbWF4ICjCsEMpIikgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDMwMCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSksCiAgICAgICAgbGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQoKc2l6ZV9mZWF0dXJlID0gZnVsbF9kYXRhICU+JSAgCiAgbXV0YXRlKGRveSA9IHlkYXkoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogIGZpbHRlcihzcF9uYW1lICVpbiUgYygiU2tpc3RvZGlhcHRvbXVzIG9yZWdvbmVuc2lzIiwgIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMiLCAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyIsICJFcGlzY2h1cmEgbGFjdXN0cmlzIikpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBzaXplLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBmYWNldF9ncmlkKHNwX25hbWV+c2V4KSArIAogIGdlb21fcG9pbnQoKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIGxhYnMoeCA9ICJEYXkgb2YgdGhlIFllYXIiLCAKICAgICAgIHkgPSAiU2l6ZSAobW0pIikgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDMwMCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSksCiAgICAgICAgbGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQoKZ2dhcnJhbmdlKGN0bWF4X2ZlYXR1cmUsIHNpemVfZmVhdHVyZSwgY29tbW9uLmxlZ2VuZCA9IFQsIGxlZ2VuZCA9ICJub25lIikKYGBgCgpgYGB7ciBzcC1wcm9wcywgZmlnLndpZHRoID0gMTIsIGZpZy5oZWlnaHQgPSA1fQphZHVsdF9zdW1tYXJpZXMgJT4lIAogIHVuZ3JvdXAoKSAlPiUgCiAgbXV0YXRlKGNvbGxlY3Rpb25fbnVtID0gYXMubnVtZXJpYyhmYWN0b3IoY29sbGVjdGlvbl9kYXRlKSkpICU+JSAKICBncm91cF9ieShjb2xsZWN0aW9uX2RhdGUpICU+JSAgCiAgYXJyYW5nZShjb2xsZWN0aW9uX2RhdGUpICU+JSAKICBzZWxlY3Qoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX251bSwgc2FtcGxlX3NpemUpICU+JSAKICBtdXRhdGUoc2FtcGxlX3NpemUgPSByZXBsYWNlX25hKHNhbXBsZV9zaXplLCAwKSkgJT4lIAogIG11dGF0ZSh0b3RhbCA9IHN1bShzYW1wbGVfc2l6ZSksCiAgICAgICAgIHBlcmNlbnRhZ2UgPSBzYW1wbGVfc2l6ZSAvIHRvdGFsLAogICAgICAgICBjb2xsZWN0aW9uX2RhdGUgPSBsdWJyaWRhdGU6OmFzX2RhdGUoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogIGdncGxvdChhZXMoeCA9IGNvbGxlY3Rpb25fZGF0ZSwgeSA9IHBlcmNlbnRhZ2UsIGZpbGwgPSBzcF9uYW1lKSkgKyAKICBnZW9tX2FyZWEoKSArIAogIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBzY2FsZV95X2NvbnRpbnVvdXMoYnJlYWtzID0gYygwLDEpKSArIAogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIERhdGUiLCAKICAgICAgIHkgPSAiUHJvcG9ydGlvbiIsIAogICAgICAgZmlsbCA9ICJTcGVjaWVzIikgKyAKICB0aGVtZV9taW5pbWFsKGJhc2Vfc2l6ZSA9IDIwKSArIAogIHRoZW1lKHBhbmVsLmdyaWQgPSBlbGVtZW50X2JsYW5rKCksCiAgICAgICAgYXhpcy50aWNrcyA9IGVsZW1lbnRfbGluZSgpKQpgYGAKCgojIyBUZW1wZXJhdHVyZSBWYXJpYWJpbGl0eQpMYWtlIENoYW1wbGFpbiBpcyBoaWdobHkgc2Vhc29uYWwsIHdpdGggYm90aCBhdmVyYWdlIHRlbXBlcmF0dXJlcyBhbmQgdGVtcGVyYXR1cmUgdmFyaWFiaWxpdHkgY2hhbmdpbmcgdGhyb3VnaG91dCB0aGUgeWVhci4gVGhlc2UgcGF0dGVybnMgaW4gdGhlIGV4cGVyaWVuY2VkIHRoZXJtYWwgZW52aXJvbm1lbnQgbWF5IGRyaXZlIHRoZSBvYnNlcnZlZCB2YXJpYXRpb24gaW4gY29wZXBvZCB0aGVybWFsIGxpbWl0cy4gSG93ZXZlciwgdGhlIHRpbWUgcGVyaW9kIGFmZmVjdGluZyBjb3BlcG9kIHRoZXJtYWwgbGltaXRzIGlzIHVua25vd24uIERlcGVuZGluZyB0aGUgb24gdGhlIGR1cmF0aW9uIG9mIHRpbWUgY29uc2lkZXJlZCwgdGhlcmUgYXJlIGxhcmdlIGNoYW5nZXMgaW4gdGhlIGV4cGVyaWVuY2VkIGVudmlyb25tZW50LCBpbiBwYXJ0aWN1bGFyIHJlZ2FyZGluZyB0aGUgdGVtcGVyYXR1cmUgcmFuZ2UgYW5kIHZhcmlhbmNlLiBDb25zaWRlciBmb3IgZXhhbXBsZSB0aHJlZSB0aW1lIHBlcmlvZHM6IHRoZSBkYXkgb2YgY29sbGVjdGlvbiwgb25lIHdlZWsgcHJpb3IgdG8gY29sbGVjdGlvbiwgYW5kIGZvdXIgd2Vla3MgcHJpb3IgdG8gY29sbGVjdGlvbi4gV2hpbGUgdGhlIG92ZXJhbGwgcGF0dGVybiBpcyBzaW1pbGFyLCB3ZSBjYW4gc2VlIHRoYXQsIHVuc3VycHJpc2luZ2x5LCBjb25zaWRlcmluZyBsb25nZXIgcGVyaW9kcyBvZiB0aW1lIHJlc3VsdHMgaW4gbGFyZ2VyIHJhbmdlcyBhbmQgc2xpZ2h0bHkgY2hhbmdlcyB0aGUgcGF0dGVybiBvZiB2YXJpYW5jZSBleHBlcmllbmNlZC4gCgpgYGB7ciBkYWlseS10ZW1wLWRhdGF9CiMjIERhaWx5IHZhbHVlcwpkYWlseV90ZW1wX2RhdGEgPSB0ZW1wX2RhdGEgJT4lCiAgdW5ncm91cCgpICU+JSAKICBncm91cF9ieShkYXRlKSAlPiUgCiAgc3VtbWFyaXNlKG1lYW5fdGVtcCA9IG1lYW4odGVtcCksCiAgICAgICAgICAgIG1lZF90ZW1wID0gbWVkaWFuKHRlbXApLAogICAgICAgICAgICB2YXJfdGVtcCA9IHZhcih0ZW1wKSwgCiAgICAgICAgICAgIG1pbl90ZW1wID0gbWluKHRlbXApLCAKICAgICAgICAgICAgbWF4X3RlbXAgPSBtYXgodGVtcCkpICU+JSAKICBtdXRhdGUoInJhbmdlX3RlbXAiID0gbWF4X3RlbXAgLSBtaW5fdGVtcCkKCmRheV9wcmlvcl90ZW1wX2RhdGEgPSB0ZW1wX2RhdGEgJT4lIAogIHVuZ3JvdXAoKSAlPiUgCiAgZ3JvdXBfYnkoZGF0ZSkgJT4lIAogIHN1bW1hcmlzZShtZWFuX3RlbXAgPSBtZWFuKHRlbXApLAogICAgICAgICAgICBtZWRfdGVtcCA9IG1lZGlhbih0ZW1wKSwKICAgICAgICAgICAgdmFyX3RlbXAgPSB2YXIodGVtcCksIAogICAgICAgICAgICBtaW5fdGVtcCA9IG1pbih0ZW1wKSwgCiAgICAgICAgICAgIG1heF90ZW1wID0gbWF4KHRlbXApKSAlPiUgCiAgbXV0YXRlKGRhdGUgPSBkYXRlICsgMSkgJT4lIAogIHJlbmFtZV93aXRoKC5mbiA9IH4gcGFzdGUwKCJwcmlvcl9kYXlfIiwgLngpLCAuY29scyA9IGMoLWRhdGUpKQoKZGFpbHlfcGxvdCA9IGRhaWx5X3RlbXBfZGF0YSAlPiUgCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKC1kYXRlKSwKICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ0ZW1wIikgJT4lIAogIGdncGxvdChhZXMoeCA9IGRhdGUsIHkgPSB0ZW1wLCBjb2xvdXIgPSBwYXJhbWV0ZXIpKSArIAogIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygKICAgICJtZWFuX3RlbXAiID0gIm9saXZlZHJhYjMiLAogICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAogICAgIm1heF90ZW1wIiA9ICJ0b21hdG8iLCAgCiAgICAibWluX3RlbXAiID0gImRvZGdlcmJsdWUiLAogICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAogICAgInZhcl90ZW1wIiA9ICJkYXJrZ29sZGVucm9kMSIKICApKSArIAogIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArIAogIGdndGl0bGUoIkRhaWx5IFZhbHVlcyIpICsgCiAgbGFicyh5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwKICAgICAgIHggPSAiIikgKyAKICB0aGVtZV9idyhiYXNlX3NpemUgPSAyMCkgKyAKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKYGBgCgpgYGB7ciBwcmVkaWN0b3JzLWZ1bmN0aW9ufQojIyBEZWZpbmluZyB0aGUgZnVuY3Rpb24gdG8gZ2V0IHByZWRpY3RvciB2YWx1ZXMgZm9yIHBlcmlvZHMgb2YgZGlmZmVyZW50IGxlbmd0aHMKZ2V0X3ByZWRpY3RvcnMgPSBmdW5jdGlvbihkYWlseV92YWx1ZXMsIHJhd190ZW1wLCBuX2RheXMpewogIHByZWZpeCA9IHN0cl9yZXBsYWNlX2FsbCh4ZnVuOjpudW1iZXJzX3RvX3dvcmRzKG5fZGF5cyksIHBhdHRlcm4gPSAiICIsIHJlcGxhY2VtZW50ID0gIi0iKQogIAogIG1lYW5fdmFsdWVzID0gZGFpbHlfdmFsdWVzICU+JSAKICAgIHVuZ3JvdXAoKSAlPiUgCiAgICBtdXRhdGUobWVhbl9tYXggPSBzbGlkZV92ZWMoLnggPSBtYXhfdGVtcCwgLmYgPSBtZWFuLCAuYmVmb3JlID0gbl9kYXlzLCAuY29tcGxldGUgPSBUKSwKICAgICAgICAgICBtZWFuX21pbiA9IHNsaWRlX3ZlYygueCA9IG1pbl90ZW1wLCAuZiA9IG1lYW4sIC5iZWZvcmUgPSBuX2RheXMsIC5jb21wbGV0ZSA9IFQpLAogICAgICAgICAgIG1lYW5fcmFuZ2UgPSBzbGlkZV92ZWMoLnggPSByYW5nZV90ZW1wLCAuZiA9IG1lYW4sIC5iZWZvcmUgPSBuX2RheXMsIC5jb21wbGV0ZSA9IFQpKSAlPiUgCiAgICBzZWxlY3QoZGF0ZSwgbWVhbl9tYXgsIG1lYW5fbWluLCBtZWFuX3JhbmdlKSAlPiUgCiAgICByZW5hbWVfd2l0aCggfiBwYXN0ZShwcmVmaXgsICJkYXkiLCAueCwgc2VwID0gIl8iKSwgLmNvbHMgPSBjKC1kYXRlKSkKICAKICBwZXJpb2RfdmFsdWVzID0gcmF3X3RlbXAgJT4lIAogICAgbXV0YXRlKG1lYW4gPSBzbGlkZV9pbmRleF9tZWFuKHRlbXAsIGkgPSBkYXRlLCBiZWZvcmUgPSBkYXlzKG5fZGF5cyksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5hX3JtID0gVCksCiAgICAgICAgICAgbWF4ID0gc2xpZGVfaW5kZXhfbWF4KHRlbXAsIGkgPSBkYXRlLCBiZWZvcmUgPSBkYXlzKG5fZGF5cyksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYV9ybSA9IFQpLAogICAgICAgICAgIG1pbiA9IHNsaWRlX2luZGV4X21pbih0ZW1wLCBpID0gZGF0ZSwgYmVmb3JlID0gZGF5cyhuX2RheXMpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYV9ybSA9IFQpLAogICAgICAgICAgIG1lZCA9IHNsaWRlX2luZGV4X2RibCh0ZW1wLCAuaSA9IGRhdGUsIC5iZWZvcmUgPSBkYXlzKG5fZGF5cyksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYV9ybSA9IFQsIC5mID0gbWVkaWFuKSwKICAgICAgICAgICB2YXIgPSBzbGlkZV9pbmRleF9kYmwodGVtcCwgLmkgPSBkYXRlLCAuYmVmb3JlID0gZGF5cyhuX2RheXMpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLmYgPSB2YXIpLAogICAgICAgICAgIHJhbmdlID0gbWF4IC0gbWluKSAlPiUgIAogICAgc2VsZWN0KC10ZW1wKSAlPiUgIAogICAgZGlzdGluY3QoKSAlPiUgCiAgICByZW5hbWVfd2l0aCggfiBwYXN0ZShwcmVmaXgsICJkYXkiLCAueCwgc2VwID0gIl8iKSwgLmNvbHMgPSBjKC1kYXRlKSklPiUgCiAgICBpbm5lcl9qb2luKG1lYW5fdmFsdWVzLCBieSA9IGMoImRhdGUiKSkgJT4lICAKICAgIGRyb3BfbmEoKQogIAogIHJldHVybihwZXJpb2RfdmFsdWVzKQp9CmBgYAoKYGBge3IgcHJlZGljdG9ycy1hbmQtcGxvdHMsIGZpZy53aWR0aD0xMiwgZmlnLmhlaWdodD01fQojICMjIEdldHRpbmcgcHJlZGljdG9yIHZhcmlhYmxlcyBmb3IgZGlmZmVyZW50IHBlcmlvZHMKIyAKIyAjIyMgU2hvcnQgKHRocmVlIGRheXMpCiMgdGhyZWVfZGF5X3RlbXBzID0gZ2V0X3ByZWRpY3RvcnMoZGFpbHlfdmFsdWVzID0gZGFpbHlfdGVtcF9kYXRhLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfdGVtcCA9IHRlbXBfZGF0YSwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gMykKIyAKIyAjIyMgT05FIFdFRUsKd2Vla190ZW1wcyA9IGdldF9wcmVkaWN0b3JzKGRhaWx5X3ZhbHVlcyA9IGRhaWx5X3RlbXBfZGF0YSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd190ZW1wID0gdGVtcF9kYXRhLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gNykKCndlZWtfcGxvdCA9IHdlZWtfdGVtcHMgJT4lCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKC1kYXRlKSwKICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwKICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInRlbXAiKSAlPiUKICBmaWx0ZXIocGFyYW1ldGVyICVpbiUgYygic2V2ZW5fZGF5X21lYW4iLAogICAgICAgICAgICAgICAgICAgICAgICAgICJzZXZlbl9kYXlfbWVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAic2V2ZW5fZGF5X21heCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgInNldmVuX2RheV9taW4iLAogICAgICAgICAgICAgICAgICAgICAgICAgICJzZXZlbl9kYXlfdmFyIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAic2V2ZW5fZGF5X3JhbmdlIikpICU+JQogIG11dGF0ZShwYXJhbWV0ZXIgPSBwYXN0ZSh3b3JkKHBhcmFtZXRlciwgc3RhcnQgPSAzLCBzZXAgPSBmaXhlZCgiXyIpKSwgIl90ZW1wIiwgc2VwID0gIiIpKSAlPiUKICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKwogIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKAogICAgIm1lYW5fdGVtcCIgPSAib2xpdmVkcmFiMyIsCiAgICAibWVkX3RlbXAiID0gInNlYWdyZWVuMyIsCiAgICAibWF4X3RlbXAiID0gInRvbWF0byIsCiAgICAibWluX3RlbXAiID0gImRvZGdlcmJsdWUiLAogICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAogICAgInZhcl90ZW1wIiA9ICJkYXJrZ29sZGVucm9kMSIKICApKSArCiAgc2NhbGVfeF9jb250aW51b3VzKGJyZWFrcyA9IGFzLkRhdGUoYygiMjAyMy0wMS0wMSIsICIyMDIzLTA0LTAxIiwgIjIwMjMtMDctMDEiKSkpICsKICBnZ3RpdGxlKCJPbmUgV2VlayIpICsKICBsYWJzKHkgPSAiVGVtcGVyYXR1cmUgKMKwQykiLAogICAgICAgeCA9ICIiKSArCiAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgVFdPIFdFRUtTCiMgdHdvX3dlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSB0ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gMTQpCiMgCiMgdHdvX3dlZWtfcGxvdCA9IHR3b193ZWVrX3RlbXBzICU+JSAKIyAgIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiMgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwgCiMgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInRlbXAiKSAlPiUgCiMgICBmaWx0ZXIocGFyYW1ldGVyICVpbiUgYygiZm91cnRlZW5fZGF5X21lYW4iLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZvdXJ0ZWVuX2RheV9tZWQiLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZvdXJ0ZWVuX2RheV9tYXgiLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmb3VydGVlbl9kYXlfbWluIiwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZm91cnRlZW5fZGF5X3ZhciIsCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZm91cnRlZW5fZGF5X3JhbmdlIikpICU+JSAKIyAgIG11dGF0ZShwYXJhbWV0ZXIgPSBwYXN0ZSh3b3JkKHBhcmFtZXRlciwgc3RhcnQgPSAzLCBzZXAgPSBmaXhlZCgiXyIpKSwgIl90ZW1wIiwgc2VwID0gIiIpKSAlPiUgCiMgICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKyAKIyAgIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArIAojICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKAojICAgICAibWVhbl90ZW1wIiA9ICJvbGl2ZWRyYWIzIiwKIyAgICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAojICAgICAibWF4X3RlbXAiID0gInRvbWF0byIsICAKIyAgICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKIyAgICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAojICAgICAidmFyX3RlbXAiID0gImRhcmtnb2xkZW5yb2QxIgojICAgKSkgKyAKIyAgIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArIAojICAgZ2d0aXRsZSgiVHdvIFdlZWtzIikgKyAKIyAgIGxhYnMoeSA9ICJUZW1wZXJhdHVyZSAowrBDKSIsCiMgICAgICAgIHggPSAiIikgKyAKIyAgIHRoZW1lX2J3KGJhc2Vfc2l6ZSA9IDIwKSArIAojICAgdGhlbWUocGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSwKIyAgICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgRk9VUiBXRUVLUwpmb3VyX3dlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd190ZW1wID0gdGVtcF9kYXRhLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuX2RheXMgPSAyOCkKCmZvdXJfd2Vla19wbG90ID0gZm91cl93ZWVrX3RlbXBzICU+JQogIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInBhcmFtZXRlciIsCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ0ZW1wIikgJT4lCiAgZmlsdGVyKHBhcmFtZXRlciAlaW4lIGMoInR3ZW50eS1laWdodF9kYXlfbWVhbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgInR3ZW50eS1laWdodF9kYXlfbWVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAidHdlbnR5LWVpZ2h0X2RheV9tYXgiLAogICAgICAgICAgICAgICAgICAgICAgICAgICJ0d2VudHktZWlnaHRfZGF5X21pbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgInR3ZW50eS1laWdodF9kYXlfdmFyIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAidHdlbnR5LWVpZ2h0X2RheV9yYW5nZSIpKSAlPiUKICBtdXRhdGUocGFyYW1ldGVyID0gcGFzdGUod29yZChwYXJhbWV0ZXIsIHN0YXJ0ID0gMywgc2VwID0gZml4ZWQoIl8iKSksICJfdGVtcCIsIHNlcCA9ICIiKSkgJT4lCiAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IHRlbXAsIGNvbG91ciA9IHBhcmFtZXRlcikpICsKICBnZW9tX2xpbmUobGluZXdpZHRoID0gMSkgKwogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygKICAgICJtZWFuX3RlbXAiID0gIm9saXZlZHJhYjMiLAogICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAogICAgIm1heF90ZW1wIiA9ICJ0b21hdG8iLAogICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKICAgICJyYW5nZV90ZW1wIiA9ICJnb2xkZW5yb2QzIiwKICAgICJ2YXJfdGVtcCIgPSAiZGFya2dvbGRlbnJvZDEiCiAgKSkgKwogIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArCiAgZ2d0aXRsZSgiRm91ciBXZWVrcyIpICsKICBsYWJzKHkgPSAiVGVtcGVyYXR1cmUgKMKwQykiLAogICAgICAgeCA9ICIiKSArCiAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgRUlHSFQgV0VFS1MKIyBlaWdodF93ZWVrX3RlbXBzID0gZ2V0X3ByZWRpY3RvcnMoZGFpbHlfdmFsdWVzID0gZGFpbHlfdGVtcF9kYXRhLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSB0ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuX2RheXMgPSA1NikKIyAKIyBlaWdodF93ZWVrX3Bsb3QgPSBlaWdodF93ZWVrX3RlbXBzICU+JSAKIyAgIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiMgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwgCiMgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInRlbXAiKSAlPiUgCiMgICBmaWx0ZXIocGFyYW1ldGVyICVpbiUgYygiZmlmdHktc2l4X2RheV9tZWFuIiwKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmaWZ0eS1zaXhfZGF5X21lZCIsCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZmlmdHktc2l4X2RheV9tYXgiLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmaWZ0eS1zaXhfZGF5X21pbiIsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZpZnR5LXNpeF9kYXlfdmFyIiwKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmaWZ0eS1zaXhfZGF5X3JhbmdlIikpICU+JSAKIyAgIG11dGF0ZShwYXJhbWV0ZXIgPSBwYXN0ZSh3b3JkKHBhcmFtZXRlciwgc3RhcnQgPSAzLCBzZXAgPSBmaXhlZCgiXyIpKSwgIl90ZW1wIiwgc2VwID0gIiIpKSAlPiUgCiMgICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKyAKIyAgIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArIAojICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKAojICAgICAibWVhbl90ZW1wIiA9ICJvbGl2ZWRyYWIzIiwKIyAgICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAojICAgICAibWF4X3RlbXAiID0gInRvbWF0byIsICAKIyAgICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKIyAgICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAojICAgICAidmFyX3RlbXAiID0gImRhcmtnb2xkZW5yb2QxIgojICAgKSkgKyAKIyAgIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArIAojICAgZ2d0aXRsZSgiRWlnaHQgV2Vla3MiKSArIAojICAgbGFicyh5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwKIyAgICAgICAgeCA9ICIiKSArIAojICAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsgCiMgICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAojICAgICAgICAgYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSAyNzAsIGhqdXN0ID0gMCwgdmp1c3QgPSAwLjUpKQojIApnZ2FycmFuZ2UoZGFpbHlfcGxvdCwgd2Vla19wbG90LCBmb3VyX3dlZWtfcGxvdCwgCiAgICAgICAgICBjb21tb24ubGVnZW5kID0gVCwgbnJvdyA9IDEsIGxlZ2VuZCA9ICJib3R0b20iKQpgYGAKClRoZSBkaWZmZXJlbnQgdGltZSBwZXJpb2RzIGV4YW1pbmVkIGJ5IHRoaXMgY2xpbWF0ZSBkYXRhIGhpZ2hsaWdodHMgdGhhdCB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gbWluaW11bSBhbmQgbWF4aW11bSB0ZW1wZXJhdHVyZXMgY2hhbmdlcyBiYXNlZCBvbiB0aGUgd2luZG93IGV4YW1pbmVkLiBGb3IgZXhhbXBsZSwgbWluaW11bSBhbmQgbWF4aW11bSB0ZW1wZXJhdHVyZXMgZXhwZXJpZW5jZWQgb3ZlciB3ZWVrbHkgaW50ZXJ2YWxzIGFyZSBjbG9zZWx5IGxpbmtlZCwgd2hlcmVhcyB0aGVyZSBpcyBhIGRpc3RpbmN0IHNlYXNvbmFsIGN5Y2xlIGluIHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiBtaW5pbXVtIGFuZCBtYXhpbXVtIHRlbXBlcmF0dXJlcyBleHBlcmllbmNlZCBvdmVyIHBlcmlvZHMgb2YgZm91ciB3ZWVrcy4gCgpgYGB7cn0Kb25lX3dlZWtfZG95X2RhdGEgPSB3ZWVrX3RlbXBzICU+JSAKICBtdXRhdGUoZG95ID0geWRheShkYXRlKSkKCm9uZV93ZWVrX3RlbXBfY2lyY2xlID0gZ2dwbG90KG9uZV93ZWVrX2RveV9kYXRhLCBhZXMoeCA9IHNldmVuX2RheV9tZWFuX21heCwgeSA9IHNldmVuX2RheV9tZWFuX21pbiwgY29sb3VyID0gZG95KSkgKyAKICBnZW9tX3BvaW50KCkgKyAKICBzY2FsZV9jb2xvdXJfZ3JhZGllbnQyKAogICAgaGlnaCA9ICJkb2RnZXJibHVlNCIsCiAgICBtaWQgPSAiY29yYWwyIiwKICAgIGxvdyA9ICJkb2RnZXJibHVlNCIsCiAgICBtaWRwb2ludCA9IDE4Mi41KSArIAogIGxhYnMoeCA9ICJNYXguIFRlbXAuICjCsEMpIiwKICAgICAgIHkgPSAiTWluLiBUZW1wLiAowrBDKSIpICsgCiAgbGFicyh4ID0gIk1heC4gVGVtcC4gKMKwQykiLAogICAgICAgeSA9ICJNaW4uIFRlbXAuICjCsEMpIikgKyAKICBnZ3RpdGxlKCJPbmUgV2VlayIpICsgCiAgdGhlbWVfbWF0dCgpCgpmb3VyX3dlZWtfZG95X2RhdGEgPSBmb3VyX3dlZWtfdGVtcHMgJT4lIAogIG11dGF0ZShkb3kgPSB5ZGF5KGRhdGUpKQoKZm91cl93ZWVrX3RlbXBfY2lyY2xlID0gZ2dwbG90KGZvdXJfd2Vla19kb3lfZGF0YSwgYWVzKHggPSBgdHdlbnR5LWVpZ2h0X2RheV9tYXhgLCB5ID0gYHR3ZW50eS1laWdodF9kYXlfbWluYCwgY29sb3VyID0gZG95KSkgKyAKICBnZW9tX3BvaW50KCkgKyAKICBzY2FsZV9jb2xvdXJfZ3JhZGllbnQyKAogICAgaGlnaCA9ICJkb2RnZXJibHVlNCIsCiAgICBtaWQgPSAiY29yYWwyIiwKICAgIGxvdyA9ICJkb2RnZXJibHVlNCIsCiAgICBtaWRwb2ludCA9IDE4Mi41KSArIAogIGxhYnMoeCA9ICJNYXguIFRlbXAuICjCsEMpIiwKICAgICAgIHkgPSAiTWluLiBUZW1wLiAowrBDKSIpICsgCiAgZ2d0aXRsZSgiRm91ciBXZWVrIikgKyAKICB0aGVtZV9tYXR0KCkKCmdnYXJyYW5nZShvbmVfd2Vla190ZW1wX2NpcmNsZSwgZm91cl93ZWVrX3RlbXBfY2lyY2xlLAogICAgICAgICAgY29tbW9uLmxlZ2VuZCA9IFQsIGxlZ2VuZCA9ICJib3R0b20iKQpgYGAKClRoZSB0aGVybWFsIGVudmlyb25tZW50IG92ZXIgYW55IHBlcmlvZCBvZiB0aW1lIG1heSBkcml2ZSBwYXR0ZXJucyBpbiB0aGVybWFsIGFjY2xpbWF0aW9uLiBUbyBleHBsb3JlIHRoZSBwb3RlbnRpYWwgZWZmZWN0cyBvZiBkaWZmZXJlbnQgYWNjbGltYXRpb24gd2luZG93cywgd2UgZXhhbWluZWQgdGhlIGNvcnJlbGF0aW9uIGJldHdlZW4gdGhlcm1hbCBsaW1pdHMgYW5kIGRpZmZlcmVudCByZXByZXNlbnRhdGlvbnMgb2YgdGhlIHRoZXJtYWwgZW52aXJvbm1lbnQgZm9yIGRpZmZlcmVudCBwZXJpb2RzIG9mIHRpbWUuIFNob3duIGJlbG93IGFyZSB0aGUgY29ycmVsYXRpb24gY29lZmZpY2llbnRzIGZvciB0aGVzZSByZWxhdGlvbnNoaXBzLiBFYWNoIGZhY2V0IHNob3dzIHRoZSByZWxhdGlvbnNoaXAgZm9yIGEgZGlmZmVyZW50IGRpbWVuc2lvbiBvZiB0aGUgdGhlcm1hbCBlbnZpcm9ubWVudC4gQ29ycmVsYXRpb24gY29lZmZpY2llbnRzIGFyZSBwbG90dGVkIGZvciBkaWZmZXJlbnQgZHVyYXRpb25zLCBmb3Igc3BlY2llcyB0aGF0IHdlcmUgY29sbGVjdGVkIG1vcmUgdGhhbiBmaXZlIHRpbWVzLiBPbmx5IGRhdGEgZm9yIG1hdHVyZSBmZW1hbGUgY29wZXBvZHMgd2FzIGluY2x1ZGVkLiAKCldlIGNhbiBzZWUgdGhhdCwgaW4gZ2VuZXJhbCwgY29wZXBvZHMgYXJlIHJlc3BvbmRpbmcgdG8gcHJveGltYXRlIGN1ZXMgZnJvbSB0aGUgdGhlcm1hbCBlbnZpcm9ubWVudCwgd2l0aCBjb3JyZWxhdGlvbnMgZ2VuZXJhbGx5IGRyb3BwaW5nIG9mZiBzdWJzdGFudGlhbGx5IGFzIGFjY2xpbWF0aW9uIHdpbmRvdyBkdXJhdGlvbiBpbmNyZWFzZXMuIEFuIGV4Y2VwdGlvbiBpcyAqRXBpc2NodXJhIGxhY3VzdHJpcyosIHdoaWNoIGFwcGVhcnMgdG8gYmUgcmVzcG9uZGluZyB0byBtYXhpbXVtIHRlbXBlcmF0dXJlcyBleHBlcmllbmNlZCBvdmVyIGEgMjAgZGF5IHRpbWUgcGVyaW9kLiAKCmBgYHtyfQojIyMgUHVsbGluZyBwcmVkaWN0b3JzIGFuZCBtZWFzdXJpbmcgY29ycmVsYXRpb25zIGZvciBtdWNoIGZpbmVyIHRpbWVzY2FsZXM7IDEtNTYgZGF5cwoKbnVtX2NvbGxzID0gZnVsbF9kYXRhICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgc2VsZWN0KGNvbGxlY3Rpb25fZGF0ZSwgc3BfbmFtZSkgJT4lICAKICBkaXN0aW5jdCgpICU+JSAgCiAgY291bnQoc3BfbmFtZSkgJT4lIAogIGZpbHRlcihuID49IDUpCgpjb3JyX3ZhbHMgPSBkYXRhLmZyYW1lKCkKCmR1cl92YWxzID0gYygxOjUwKQpmb3IoaSBpbiBkdXJfdmFscyl7CiAgCiAgZHVyYXRpb25fdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSB0ZW1wX2RhdGEsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gaSkgJT4lIAogICAgZmlsdGVyKGRhdGUgJWluJSBhc19kYXRlKHVuaXF1ZShmdWxsX2RhdGEkY29sbGVjdGlvbl9kYXRlKSkpCiAgCiAgY29ycl9kYXRhID0gZnVsbF9kYXRhICU+JQogICAgZmlsdGVyKHNwX25hbWUgJWluJSBudW1fY29sbHMkc3BfbmFtZSkgJT4lIAogICAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogICAgbXV0YXRlKGNvbGxlY3Rpb25fZGF0ZSA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogICAgaW5uZXJfam9pbihkdXJhdGlvbl90ZW1wcywgam9pbl9ieShjb2xsZWN0aW9uX2RhdGUgPT0gZGF0ZSkpICU+JSAKICAgIHBpdm90X2xvbmdlcihjb2xzID0gYyhjb2xsZWN0aW9uX3RlbXAsIGNvbnRhaW5zKCJkYXlfIikpLAogICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ2YWx1ZSIsIAogICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInByZWRpY3RvciIpICU+JSAgCiAgICBncm91cF9ieShzcF9uYW1lLCBwcmVkaWN0b3IpICU+JSAKICAgIHN1bW1hcmlzZShjb3JyZWxhdGlvbiA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkZXN0aW1hdGUsCiAgICAgICAgICAgICAgcC52YWx1ZSA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkcC52YWx1ZSwKICAgICAgICAgICAgICBjaV9sb3cgPSBjb3IudGVzdChjdG1heCwgdmFsdWUpJGNvbmYuaW50WzFdLAogICAgICAgICAgICAgIGNpX2hpZ2ggPSBjb3IudGVzdChjdG1heCwgdmFsdWUpJGNvbmYuaW50WzJdLAogICAgICAgICAgICAgIC5ncm91cHMgPSAia2VlcCIpICU+JSAKICAgIGZpbHRlcihwcmVkaWN0b3IgIT0gImNvbGxlY3Rpb25fdGVtcCIpICU+JSAKICAgIG11dGF0ZShzaWcgPSBpZmVsc2UocC52YWx1ZSA8MC4wNSwgIlNpZy4iLCAiTm9uIFNpZy4iKSkgJT4lIAogICAgc2VwYXJhdGUocHJlZGljdG9yLCAiX2RheV8iLCBpbnRvID0gYyhOQSwgInBhcmFtZXRlciIpKSAlPiUgCiAgICBtdXRhdGUoZHVyYXRpb24gPSBpKQogIAogIGNvcnJfdmFscyA9IGJpbmRfcm93cyhjb3JyX3ZhbHMsIGNvcnJfZGF0YSkKfQoKY29sbF9jb3JyID0gZnVsbF9kYXRhICU+JQogIGZpbHRlcihzcF9uYW1lICVpbiUgbnVtX2NvbGxzJHNwX25hbWUpICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lIAogIHN1bW1hcmlzZShjb3JyZWxhdGlvbiA9IGNvci50ZXN0KGN0bWF4LCBjb2xsZWN0aW9uX3RlbXApJGVzdGltYXRlLAogICAgICAgICAgICBwLnZhbHVlID0gY29yLnRlc3QoY3RtYXgsIGNvbGxlY3Rpb25fdGVtcCkkcC52YWx1ZSwKICAgICAgICAgICAgY2lfbG93ID0gY29yLnRlc3QoY3RtYXgsIGNvbGxlY3Rpb25fdGVtcCkkY29uZi5pbnRbMV0sCiAgICAgICAgICAgIGNpX2hpZ2ggPSBjb3IudGVzdChjdG1heCwgY29sbGVjdGlvbl90ZW1wKSRjb25mLmludFsyXSkgJT4lIAogIG11dGF0ZShzaWcgPSBpZmVsc2UocC52YWx1ZSA8MC4wNSwgIlNpZy4iLCAiTm9uIFNpZy4iKSkgJT4lIAogIG11dGF0ZShkdXJhdGlvbiA9IDAsCiAgICAgICAgIHBhcmFtZXRlciA9ICJjb2xsX3RlbXAiKQoKY29ycl92YWxzID0gY29ycl92YWxzICU+JSAgCiAgbXV0YXRlKGR1cmF0aW9uID0gYXMubnVtZXJpYyhkdXJhdGlvbikpICU+JSAKICBiaW5kX3Jvd3MoY29sbF9jb3JyKQoKYGBgCgpgYGB7ciBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9Nn0KY29ycl92YWxzICU+JSAKICBtdXRhdGUocGFyYW1ldGVyID0gZmN0X3JlbGV2ZWwocGFyYW1ldGVyLCBjKCJtaW4iLCAibWF4IiwgInJhbmdlIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJtZWFuIiwgIm1lZCIsICJ2YXIiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIm1lYW5fbWluIiwgIm1lYW5fbWF4IiwgIm1lYW5fcmFuZ2UiKSkpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBkdXJhdGlvbiwgeSA9IGNvcnJlbGF0aW9uLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBmYWNldF93cmFwKC5+cGFyYW1ldGVyKSArIAogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDApICsgCiAgZ2VvbV9wb2ludChzaXplID0gMC45KSArIAogIGdlb21fbGluZShsaW5ld2lkdGggPSAxLjUpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgbGFicyh4ID0gIkR1cmF0aW9uIChkYXlzKSIsCiAgICAgICB5ID0gIkNvcnJlbGF0aW9uIiwgCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKQpgYGAKClNob3duIGhlcmUgYXJlIHRoZSB0b3AgdGhyZWUgZmFjdG9ycyBmb3IgZWFjaCBzcGVjaWVzLiAKCmBgYHtyIHByZWRpY3Rvci1jb3JyZWxhdGlvbnN9CiMgCiMgY29ycl92YWxzID0gZnVsbF9kYXRhICU+JQojICAgZmlsdGVyKHNwX25hbWUgJWluJSBudW1fY29sbHMkc3BfbmFtZSkgJT4lIAojICAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAojICAgbXV0YXRlKGNvbGxlY3Rpb25fZGF0ZSA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAojICAgZnVsbF9qb2luKHRlbXBfcHJlZGljdG9ycywgam9pbl9ieShjb2xsZWN0aW9uX2RhdGUgPT0gZGF0ZSkpICU+JSAKIyAgIHBpdm90X2xvbmdlcihjb2xzID0gYyhjb2xsZWN0aW9uX3RlbXAsIG1lYW5fdGVtcDp0YWlsKG5hbWVzKC4pLCAxKSksCiMgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInZhbHVlIiwgCiMgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicHJlZGljdG9yIikgJT4lICAKIyAgIGdyb3VwX2J5KHNwX25hbWUsIHByZWRpY3RvcikgJT4lIAojICAgc3VtbWFyaXNlKGNvcnJlbGF0aW9uID0gY29yLnRlc3QoY3RtYXgsIHZhbHVlKSRlc3RpbWF0ZSwKIyAgICAgICAgICAgICBwLnZhbHVlID0gY29yLnRlc3QoY3RtYXgsIHZhbHVlKSRwLnZhbHVlLAojICAgICAgICAgICAgIGNpX2xvdyA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkY29uZi5pbnRbMV0sCiMgICAgICAgICAgICAgY2lfaGlnaCA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkY29uZi5pbnRbMl0pICU+JSAKIyAgIG11dGF0ZShzaWcgPSBpZmVsc2UocC52YWx1ZSA8MC4wNSwgIlNpZy4iLCAiTm9uIFNpZy4iKSkKCmNvcnJfdmFscyAlPiUgIAogIGZpbHRlcihzaWcgPT0gIlNpZy4iKSAlPiUgCiAgZHJvcF9uYShjb3JyZWxhdGlvbikgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JQogIGFycmFuZ2UoZGVzYyhjb3JyZWxhdGlvbikpICU+JSAKICBzbGljZV9oZWFkKG4gPSAzKSAlPiUgCiAgc2VsZWN0KCJTcGVjaWVzIiA9IHNwX25hbWUsICJQcmVkaWN0b3IiID0gcGFyYW1ldGVyLCAiRHVyYXRpb24iID0gZHVyYXRpb24sICJDb3JyZWxhdGlvbiIgPSBjb3JyZWxhdGlvbiwgIlAtVmFsdWUiID0gcC52YWx1ZSkgJT4lIAogIGtuaXRyOjprYWJsZShhbGlnbiA9ICJjIikKYGBgCgpQaGVub3R5cGljIHZhcmlhdGlvbiAobGlrZSBhY2NsaW1hdGlvbiBvZiB0aGVybWFsIGxpbWl0cykgaXMgYSBwaHlzaW9sb2dpY2FsIHByb2Nlc3MuIGRlcGVuZGluZyBvbiB0aGUgbWVjaGFuaXN0aWMgdW5kZXJwaW5uaW5ncyAoY2hhbmdlcyBpbiBIU1AgZXhwcmVzc2lvbiwgZXRjLiksIHRoZSBhbW91bnQgb2YgdGltZSBpdCB0YWtlcyBmb3IgYW4gaW5kaXZpZHVhbCB0byBhY2NsaW1hdGUgbWF5IHZhcnkgYmFzZWQgb24gYm9keSBzaXplIChsYXJnZXIgc3BlY2llcywgbW9yZSBjZWxscywgbW9yZSB0aW1lIHJlcXVpcmVkIHRvIGFjY2xpbWF0ZSkuIFNob3duIGhlcmUgaXMgdGhlIGR1cmF0aW9uIG9mIHRoZSBlbnZpcm9ubWVudGFsIGFjY2xpbWF0aW9uIHdpbmRvdyB0aGUgY29wZXBvZHMgYXBwZWFyIHRvIGJlIHJlc3BvbmRpbmcgdG8uICAKCmBgYHtyIGFjYy1kdXJhdGlvbi1wbG90LCBmaWcud2lkdGg9NywgZmlnLmhlaWdodD00fQptZWFuX3NpemVzID0gZnVsbF9kYXRhICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lICAKICBzdW1tYXJpc2UobWVhbl9zaXplID0gbWVhbihzaXplLCBuYS5ybSA9IFQpKQoKY29ycl92YWxzICU+JSAKICBncm91cF9ieShzcF9uYW1lKSAlPiUgCiAgZmlsdGVyKGNvcnJlbGF0aW9uID09IG1heChjb3JyZWxhdGlvbikpICU+JSAgCiAgaW5uZXJfam9pbihtZWFuX3NpemVzLCBieSA9ICJzcF9uYW1lIikgJT4lIAogIHNlbGVjdChzcF9uYW1lLCBkdXJhdGlvbiwgbWVhbl9zaXplKSAlPiUgIAogIGdncGxvdChhZXMoeCA9IG1lYW5fc2l6ZSwgeSA9IGR1cmF0aW9uKSkgKyAKICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBzcF9uYW1lKSwgCiAgICAgICAgICAgICBzaXplID0gNCkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiTWVhbiBGZW1hbGUgU2l6ZSAobW0pIiwKICAgICAgIHkgPSAiQWNjbGltYXRpb24gRHVyYXRpb24iLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKCiMjIFRyYWl0IFZhcmlhdGlvbiAKYGBge3IgY3RtYXgtYW5kLXNpemUtc3VtLXBsb3QsIGZpZy53aWR0aD0yMCwgZmlnLmhlaWdodD01fQojIGN0bWF4X3Bsb3QgPSBmdWxsX2RhdGEgJT4lCiMgICBtdXRhdGUoICNzcF9uYW1lID0gc3RyX3JlcGxhY2Uoc3BfbmFtZSwgcGF0dGVybiA9ICIgIiwKIyAgICAgIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlcGxhY2VtZW50ID0gIlxuIiksCiMgICAgIHNwX25hbWUgPSBmY3RfcmVvcmRlcihzcF9uYW1lLCBjdG1heCwgbWVhbikpICU+JQojICAgZ2dwbG90KGFlcyh5ID0gc3BfbmFtZSwgeCA9IGN0bWF4KSkgKwojICAgZ2VvbV9wb2ludChhZXMoY29sb3VyPSBzcF9uYW1lX3N1YiksCiMgICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fZG9kZ2Uod2lkdGggPSAwLjMpLAojICAgICAgICAgICAgICBzaXplID0gNCkgKwojICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsKIyAgIHhsYWIoTlVMTCkgKwojICAgbGFicyh5ID0gIiIsCiMgICAgICAgIHggPSAiQ1RtYXggKMKwQykiLAojICAgICAgICBjb2xvdXIgPSAiR3JvdXAiKSArCiMgICB0aGVtZV9tYXR0KCkgKwojICAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQojIAojIHNpemVfcGxvdCA9IGZ1bGxfZGF0YSAlPiUKIyAgIG11dGF0ZShzcF9uYW1lID0gZmN0X3Jlb3JkZXIoc3BfbmFtZSwgY3RtYXgsIG1lYW4pKSAlPiUKIyAgIGdncGxvdChhZXMoeSA9IHNwX25hbWUsIHggPSBzaXplKSkgKwojICAgZ2VvbV9wb2ludChhZXMoY29sb3VyPSBzcF9uYW1lX3N1YiksCiMgICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fZG9kZ2Uod2lkdGggPSAwLjMpLAojICAgICAgICAgICAgICBzaXplID0gNCkgKwojICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsKIyAgIGxhYnMoeCA9ICJQcm9zb21lIExlbmd0aCAobW0pIiwKIyAgICAgICAgeSA9ICIiLAojICAgICAgICBjb2xvdXIgPSAiR3JvdXAiKSArCiMgICBndWlkZXMoY29sb3IgPSBndWlkZV9sZWdlbmQobmNvbCA9IDEpKSArCiMgICB0aGVtZV9tYXR0KGJhc2Vfc2l6ZSA9ICkgKwojICAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IiwKIyAgICAgICAgIGF4aXMudGV4dC55ID0gZWxlbWVudF9ibGFuaygpLAojICAgICAgICAgcGxvdC5tYXJnaW4gPSBtYXJnaW4oMCwgMCwgMCwgMCwiY20iKSkKIyAKIyB0cmFpdF9wbG90ID0gY3RtYXhfcGxvdCArIHNpemVfcGxvdAojIHRyYWl0X3Bsb3QKYGBgCgpTaG93biBiZWxvdyBhcmUgdGhlIGNsdXRjaCBzaXplIGRpc3RyaWJ1dGlvbnMgZm9yIHRoZSB0aHJlZSBkaWFwdG9taWlkIHNwZWNpZXMsIHdoaWNoIHByb2R1Y2UgZWdnIHNhY3MgdGhhdCBhbGxvdyBmb3IgZWFzeSBxdWFudGlmaWNhdGlvbiBvZiBmZWN1bmRpdHkuIAoKYGBge3IgZmVjdW5kaXR5LWhpc3RvZ3JhbSwgZmlnLndpZHRoPTcsIGZpZy5oZWlnaHQ9MTB9CmZ1bGxfZGF0YSAlPiUgIAogIGRyb3BfbmEoZmVjdW5kaXR5KSAlPiUgIAogIGdncGxvdChhZXMoeCA9IGZlY3VuZGl0eSwgZmlsbCA9IHNwX25hbWVfc3ViKSkgKyAKICBmYWNldF93cmFwKC5+c3BfbmFtZV9zdWIsIG5jb2wgPSAxKSArIAogIGdlb21faGlzdG9ncmFtKGJpbndpZHRoID0gMikgKyAKICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgbGFicyh4ID0gIkZlY3VuZGl0eSAoIyBFZ2dzKSIpICsKICB0aGVtZV9tYXR0X2ZhY2V0cygpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQpgYGAKCk9uZSBvZiB0aGUgbWFpbiBhaW1zIG9mIHRoaXMgcHJvamVjdCBpcyB0byBleGFtaW5lIHRoZSBwYXR0ZXJucyBhbmQgcHJvY2Vzc2VzIGRyaXZpbmcgdmFyaWF0aW9uIGluIHVwcGVyIHRoZXJtYWwgbGltaXRzIGFjcm9zcyB0aGVzZSBzcGVjaWVzIG9mIGNvcGVwb2RzLiAKCiMjIyBWYXJpYXRpb24gd2l0aCB0ZW1wZXJhdHVyZSAKCldlIGV4cGVjdCBvbmUgb2YgdGhlIHByaW1hcnkgZHJpdmVycyBvZiBjb3BlcG9kIHRoZXJtYWwgbGltaXRzIHRvIGJlIHRlbXBlcmF0dXJlLiBUaGUgY29ycmVsYXRpb24gYW5hbHlzaXMgaGFzIHNob3duIHRoYXQgdGhlIGNvcGVwb2RzIGFyZSBnZW5lcmFsbHkgKGFsdGhvdWdoIG5vdCBhbHdheXMpIHJlc3BvbmRpbmcgdG8gdGhlIHJlY2VudCB0aGVybWFsIGVudmlyb25tZW50LiBTaG93biBiZWxvdyBhcmUgdGhlcm1hbCBsaW1pdHMsIGJvZHkgc2l6ZSwgYW5kIGZlY3VuZGl0eSB2YWx1ZXMgcGxvdHRlZCBhZ2FpbnN0IHRoZSB0ZW1wZXJhdHVyZSBhdCB0aGUgdGltZSBvZiBjb2xsZWN0aW9uLiBBbHNvIHNob3duIGlzIHdhcm1pbmcgdG9sZXJhbmNlLCBjYWxjdWxhdGVkIGFzIHRoZSBkaWZmZXJlbmNlIGJldHdlZW4gdXBwZXIgdGhlcm1hbCBsaW1pdCBhbmQgdGhlIGNvbGxlY3Rpb24gdGVtcGVyYXR1cmUuIAoKV2UgZ2VuZXJhbGx5IHNlZSBhbiBpbmNyZWFzZSBpbiB0aGVybWFsIGxpbWl0cyB3aXRoIGluY3JlYXNpbmcgY29sbGVjdGlvbiB0ZW1wZXJhdHVyZSwgYSBzbGlnaHQgZGVjcmVhc2UgaW4gYm9keSBzaXplLCBhbmQgdmFyaWFibGUgcmVsYXRpb25zaGlwcyBiZXR3ZWVuIHRlbXBlcmF0dXJlIGFuZCBmZWN1bmRpdHkuIEFsbCBzcGVjaWVzIG1haW50YWluZWQgc29tZSBkZWdyZWUgb2YgYnVmZmVyIGJldHdlZW4gZW52aXJvbm1lbnRhbCB0ZW1wZXJhdHVyZXMgYW5kIHVwcGVyIHRoZXJtYWwgbGltaXRzLCBidXQgKkVwaXNjaHVyYSogYW5kICpMLiBtaW51dHVzKiBhcHByb2FjaGVkIHRoZWlyIHVwcGVyIHRoZXJtYWwgbGltaXRzIGR1cmluZyB0aGUgd2FybWVzdCBjb2xsZWN0aW9ucyBkdXJpbmcgdGhlIHN1bW1lci4gCgpgYGB7ciB0cmFpdC1jb2xsLXRlbXAtcGxvdHMsIGZpZy53aWR0aD0xNSwgZmlnLmhlaWdodD0xMH0KY3RtYXhfdGVtcCA9IGdncGxvdChmdWxsX2RhdGEsIGFlcyh4ID0gY29sbGVjdGlvbl90ZW1wLCB5ID0gY3RtYXgsIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIGxpbmV3aWR0aCA9IDMpICsKICBnZW9tX3BvaW50KHNpemUgPSAzKSArIAogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIFRlbXBlcmF0dXJlICjCsEMpIiwgCiAgICAgICB5ID0gIkNUbWF4ICjCsEMpIiwKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQoKc2l6ZV90ZW1wID0gZ2dwbG90KGZpbHRlcihmdWxsX2RhdGEsIHNleCAhPSAianV2ZW5pbGUiKSwgYWVzKHggPSBjb2xsZWN0aW9uX3RlbXAsIHkgPSBzaXplLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBsaW5ld2lkdGggPSAzKSArCiAgZ2VvbV9wb2ludChzaXplID0gMykgKyAKICBsYWJzKHggPSAiQ29sbGVjdGlvbiBUZW1wZXJhdHVyZSAowrBDKSIsIAogICAgICAgeSA9ICJMZW5ndGggKG1tKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCgp3dF90ZW1wID0gZ2dwbG90KGZ1bGxfZGF0YSwgYWVzKHggPSBjb2xsZWN0aW9uX3RlbXAsIHkgPSB3YXJtaW5nX3RvbCwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgbGluZXdpZHRoID0gMykgKwogIGdlb21fcG9pbnQoc2l6ZSA9IDMpICsgCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gVGVtcGVyYXR1cmUgKMKwQykiLCAKICAgICAgIHkgPSAiV2FybWluZyBUb2xlcmFuY2UgKMKwQykiLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSAgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQoKZWdnc190ZW1wID0gZ2dwbG90KGZ1bGxfZGF0YSwgYWVzKHggPSBjb2xsZWN0aW9uX3RlbXAsIHkgPSBmZWN1bmRpdHksIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIGxpbmV3aWR0aCA9IDMpICsKICBnZW9tX3BvaW50KHNpemUgPSAzKSArIAogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIFRlbXBlcmF0dXJlICjCsEMpIiwgCiAgICAgICB5ID0gIkZlY3VuZGl0eSAoIyBFZ2dzKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCgpnZ2FycmFuZ2UoY3RtYXhfdGVtcCwgc2l6ZV90ZW1wLCB3dF90ZW1wLCBlZ2dzX3RlbXAsIAogICAgICAgICAgY29tbW9uLmxlZ2VuZCA9IFQsIGxlZ2VuZCA9ICJyaWdodCIpCmBgYAoKYGBge3IgZmlnLndpZHRoPTcsIGZpZy5oZWlnaHQ9NX0KZnVsbF9kYXRhICU+JSAKICAjZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lICAKICBncm91cF9ieShzcF9uYW1lKSAlPiUgZmlsdGVyKG4oKSA+IDUpICU+JSBmaWx0ZXIoIXN0cl9kZXRlY3Qoc3BfbmFtZSwgcGF0dGVybiA9ICJraW5kdGkiKSkgJT4lIAogIGdncGxvdChhZXMoeCA9IGNvbGxlY3Rpb25fdGVtcCwgeSA9IGN0bWF4LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBmYWNldF93cmFwKHNwX25hbWV+LikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuOCkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDIpICsgCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gVGVtcC4gKMKwQykiLCAKICAgICAgIHkgPSAiQ1RtYXggKMKwQykiKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpUZW1wZXJhdHVyZSBkZXBlbmRlbmNlIGlzIHJlbGF0aXZlbHkgd2VhayBpbiAqTC4gc2ljaWxpcyosIGVzcGVjaWFsbHkgYXQgY29vbGVyIHRlbXBlcmF0dXJlcy4gV2Ugd2lsbCByZXR1cm4gdG8gdGhpcyBmZWF0dXJlIGxhdGVyIGluIHRoZSByZXBvcnQsIGJ1dCBmb3Igbm93IHdlIHdpbGwgbm90ZSB0aGF0IHRoZXJlIGFyZSB0d28gc2l6ZSBtb3JwaHMgaW4gdGhpcyBzcGVjaWVzLCB3aGljaCBhcHBlYXIgdG8gcmVzcG9uZCBkaWZmZXJlbnRseSB0byBkZWNyZWFzZXMgaW4gdGVtcGVyYXR1cmUuIFRoZXJlIGFyZSBzaWduaWZpY2FudCBkaWZmZXJlbmNlcyBiZXR3ZWVuIHRoZSBtb3JwaHMgYW5kIGhvdyB0ZW1wZXJhdHVyZSBhZmZlY3RzIENUbWF4LiAKCmBgYHtyfQptb3JwaF9kYXRhID0gZnVsbF9kYXRhICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiICYgc3BfbmFtZSA9PSAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyIpICU+JSAKICBtdXRhdGUobW9ycGggPSBpZl9lbHNlKHNpemUgPiAwLjg5LCAibGFyZ2UiLCAic21hbGwiKSkKCmdncGxvdChtb3JwaF9kYXRhLCBhZXMoeCA9IGNvbGxlY3Rpb25fdGVtcCwgeSA9IGN0bWF4LCBjb2xvdXIgPSBtb3JwaCkpICsgCiAgZ2VvbV9wb2ludChzaXplID0gMiwgYWxwaGEgPSAwLjgpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBULCBsaW5ld2lkdGggPSAyKSArIAogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIFRlbXAuICjCsEMpIiwgCiAgICAgICB5ID0gIkNUbWF4ICjCsEMpIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCgptb3JwaC5tb2RlbCA9IGxtKGRhdGEgPSBtb3JwaF9kYXRhLCAKICAgICAgICAgICAgICAgICBjdG1heCB+IGNvbGxlY3Rpb25fdGVtcCAqIG1vcnBoKQoKa25pdHI6OmthYmxlKGNhcjo6QW5vdmEobW9ycGgubW9kZWwsIHR5cGUgPSAiSUlJIiwgdGVzdCA9ICJGIikpCgojc3VtbWFyeShtb3JwaC5tb2RlbCkKCiNtb3JwaC5lbSA9IGVtbWVhbnM6OmVtbWVhbnMobW9ycGgubW9kZWwsIHBhaXJ3aXNlIH4gbW9ycGgpCgojcGxvdChtb3JwaC5lbSkKYGBgCgpgYGB7ciBjdG1heC1yYW5nZS1wbG90LCBpbmNsdWRlID0gRn0KZnVsbF9kYXRhICU+JSAgCiAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSwgY29sbGVjdGlvbl90ZW1wKSAlPiUgIAogIHN1bW1hcmlzZSgiY3RtYXhfcmFuZ2UiID0gbWF4KGN0bWF4KSAtIG1pbihjdG1heCksCiAgICAgICAgICAgICJjdG1heF92YXIiID0gdmFyKGN0bWF4KSwKICAgICAgICAgICAgInNhbXBsZV9zaXplIiA9IG4oKSkgJT4lIAogIHVuZ3JvdXAoKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lIAogIGZpbHRlcihzcF9uYW1lICE9ICJMZXB0b2RvcmEga2luZHRpIikgJT4lIAogIGZpbHRlcihzYW1wbGVfc2l6ZSA+IDMpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBjb2xsZWN0aW9uX3RlbXAsIHkgPSBjdG1heF92YXIsIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGZhY2V0X3dyYXAoc3BfbmFtZX4uKSArIAogIGdlb21fcG9pbnQoKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIGNvbG91ciA9ICJibGFjayIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKQpgYGAKCgpgYGB7ciBjdG1heC1jb2xsLXRlbXAtbW9kZWwsIGluY2x1ZGUgPSBGfQojIGFkdWx0X2RhdGEgPSBmdWxsX2RhdGEgJT4lIAojICAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikKbW9kZWxfZGF0YSA9IGZ1bGxfZGF0YSAlPiUgIAogIGRyb3BfbmEoc2l6ZSwgY3RtYXgpICU+JSAgCiAgbXV0YXRlKHRlbXBfY2VudCA9IHNjYWxlKGNvbGxlY3Rpb25fdGVtcCwgY2VudGVyID0gVCwgc2NhbGUgPSBGKSwKICAgICAgICAgc2l6ZV9jZW50ID0gc2NhbGUoc2l6ZSwgY2VudGVyID0gVCwgc2NhbGUgPSBGKSkKCmN0bWF4X3RlbXAubW9kZWwgPSBsbShkYXRhID0gbW9kZWxfZGF0YSwgY3RtYXggfiB0ZW1wX2NlbnQgKiBzcF9uYW1lKQpzaXplX3RlbXAubW9kZWwgPSBsbShkYXRhID0gbW9kZWxfZGF0YSwgc2l6ZSB+IGNvbGxlY3Rpb25fdGVtcCAqIHNwX25hbWUpCgprbml0cjo6a2FibGUoY2FyOjpBbm92YShjdG1heF90ZW1wLm1vZGVsKSkKCmN0bWF4X3Jlc2lkcyA9IGNiaW5kKG1vZGVsX2RhdGEsICJyZXNpZHMiID0gY3RtYXhfdGVtcC5tb2RlbCRyZXNpZHVhbHMsICJzaXplX3Jlc2lkcyIgPSBzaXplX3RlbXAubW9kZWwkcmVzaWR1YWxzKQoKYGBgCgpDb3BlcG9kcyBzcGVudCBzZXZlcmFsIGRheXMgaW4gbGFiIGR1cmluZyBleHBlcmltZW50cy4gU2hvd24gYmVsb3cgYXJlIHRoZSBDVG1heCByZXNpZHVhbHMgKHRha2VuIGZyb20gYSBtb2RlbCBvZiBDVG1heCBhZ2FpbnN0IGNvbGxlY3Rpb24gdGVtcGVyYXR1cmUpIHBsb3R0ZWQgYWdhaW5zdCB0aGUgdGltZSBzcGVudCBpbiBsYWIgYmVmb3JlIG1lYXN1cmVtZW50cyB3ZXJlIG1hZGUuIEluZGl2aWR1YWwgcmVncmVzc2lvbnMgYXJlIHNob3duIGZvciB0aGUgcmVzaWR1YWxzIGFnYWluc3QgZGF5cyBpbiBsYWIgZm9yIGVhY2ggY29sbGVjdGlvbi4gV2UgY2FuIHNlZSBjbGVhcmx5IHRoYXQgdGhlcm1hbCBsaW1pdHMgYXJlIGZhaXJseSBzdGFibGUgb3ZlciB0aW1lLiAKCmBgYHtyIGN0bWF4LXRpbWUtaW4tbGFiLCBmaWcud2lkdGg9MTUsIGZpZy5oZWlnaHQ9MTB9CmdncGxvdChjdG1heF9yZXNpZHMsIGFlcyh4ID0gZGF5c19pbl9sYWIsIHkgPSByZXNpZHMsIGNvbG91ciA9IHNwX25hbWUsIGdyb3VwID0gY29sbGVjdGlvbl9kYXRlKSkgKyAKICBmYWNldF93cmFwKHNwX25hbWV+LikgKyAKICBnZW9tX3BvaW50KHNpemUgPSA0LCBhbHBoYSA9IDAuNSkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDEpICsgCiAgI3NjYWxlX3hfY29udGludW91cyhicmVha3MgPSBjKDA6NSkpICsgCiAgbGFicyh4ID0gIkRheXMgaW4gbGFiIiwgCiAgICAgICB5ID0gIkNUbWF4IFJlc2lkdWFscyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpgYGB7ciBBUlItbGltaXRzLXBsb3QsIGZpZy53aWR0aD04LCBmaWcuaGVpZ2h0PTZ9CmZ1bGwubW9kZWwgPSBsbWU0OjpsbWVyKGRhdGEgPSBtb2RlbF9kYXRhLAogICAgICAgICAgICAgICAgICAgICAgICBjdG1heCB+IHNleCArIHRlbXBfY2VudCArIHNpemVfY2VudCArCiAgICAgICAgICAgICAgICAgICAgICAgICAgKDEgKyBkYXlzX2luX2xhYiArIHRlbXBfY2VudCArIHNpemVfY2VudHxzcF9uYW1lOnNleCkpCgpjYXI6OkFub3ZhKGZ1bGwubW9kZWwpCgpmaXhlZCA9IGZpeGVmKGZ1bGwubW9kZWwpCgptb2RlbF9jb2VmcyA9IGNvZWZmaWNpZW50cyhmdWxsLm1vZGVsKSRgc3BfbmFtZTpzZXhgICU+JSAgCiAgcm93bmFtZXNfdG9fY29sdW1uKHZhciA9ICJzcGVjaWVzIikgJT4lIAogIHNlcGFyYXRlKHNwZWNpZXMsIGludG8gPSBjKCJzcGVjaWVzIiwgInNleCIpLCBzZXAgPSAiOiIpICU+JSAKICBzZWxlY3Qoc3BlY2llcywgc2V4LCAiaW50ZXJjZXB0IiA9ICIoSW50ZXJjZXB0KSIsIHRlbXBfY2VudCwgc2l6ZV9jZW50LCBkYXlzX2luX2xhYikKCmdncGxvdChtb2RlbF9jb2VmcywgYWVzKHggPSBpbnRlcmNlcHQsIHkgPSB0ZW1wX2NlbnQpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIGNvbG91ciA9ICJibGFjayIpICsKICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBzcGVjaWVzLCBzaGFwZSA9IHNleCksCiAgICAgICAgICAgICBzaXplID0gNikgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiU3BlY2llcyBJbnRlcmNlcHQiLCAKICAgICAgIHkgPSAiQVJSIikgKwogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKClRoZSB0ZXJtICJhY2NsaW1hdGlvbiByZXNwb25zZSByYXRpbyIgaXMgb2Z0ZW4gdXNlZCB0byBkZXNjcmliZSB0aGUgZWZmZWN0IG9mIHRlbXBlcmF0dXJlIG9uIHRoZXJtYWwgbGltaXRzLiBUaGUgQVJSIGlzIGNhbGN1bGF0ZWQgYXMgdGhlIGNoYW5nZSBpbiB0aGVybWFsIGxpbWl0cyBwZXIgZGVncmVlIGNoYW5nZSBpbiBhY2NsaW1hdGlvbiB0ZW1wZXJhdHVyZS4gRm9yIG91ciBkYXRhLCB3ZSB3aWxsIGVzdGltYXRlIEFSUiBhcyB0aGUgc2xvcGUgb2YgQ1RtYXggYWdhaW5zdCBjb2xsZWN0aW9uIHRlbXBlcmF0dXJlLiBUaGVzZSBzbG9wZXMgd2VyZSB0YWtlbiBmcm9tIGEgcmVncmVzc2lvbiBvZiBDVG1heCBhZ2FpbnN0IGNvbGxlY3Rpb24gdGVtcGVyYXR1cmUgYW5kIGJvZHkgc2l6ZS4gVHdvIGRpZmZlcmVudCBtb2RlbCB0eXBlcyB3ZXJlIHVzZWQsIGEgc2ltcGxlIGxpbmVhciByZWdyZXNzaW9uIGFuZCBhIG1peGVkIGVmZmVjdHMgbW9kZWwuIFRoZSBlc3RpbWF0ZWQgQVJSIHZhbHVlcyB3ZXJlIGdlbmVyYWxseSBoaWdobHkgc2ltaWxhciBiZXR3ZWVuIHRoZSBtb2RlbCB0eXBlcyB1c2VkLgoKYGBge3IgYXJyLWNvbXAtcGxvdCwgZmlnLndpZHRoPTgsIGZpZy5oZWlnaHQ9MTAsIGluY2x1ZGUgPSBGfQpjb2VmX21vZGVsX2RhdGEgPSBmdWxsX2RhdGEgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUsIHNleCkgJT4lIAogIGZpbHRlcihuKCkgPiAzICYgIXN0cl9kZXRlY3Qoc3BfbmFtZSwgcGF0dGVybiA9ICJraW5kdGkiKSkgCgpjb2VmX24gPSBmdWxsX2RhdGEgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUsIHNleCkgJT4lIAogIGZpbHRlcihuKCkgPiA1KSAlPiUgCiAgc3VtbWFyaXNlKHNhbXBsZV9uID0gbigpLCAKICAgICAgICAgICAgbWVhbl9jdG1heCA9IG1lYW4oY3RtYXgpKQoKQVJSX3ZhbHMgPSBjb2VmX21vZGVsX2RhdGEgJT4lIAogIGRvKGJyb29tOjp0aWR5KGxtKGN0bWF4IH4gY29sbGVjdGlvbl90ZW1wICsgc2l6ZSwgZGF0YSA9IC4pKSkgJT4lIAogIGZpbHRlcih0ZXJtID09ICJjb2xsZWN0aW9uX3RlbXAiKSAlPiUgCiAgc2VsZWN0KHNwX25hbWUsIHNleCwgIkFSUiIgPSBlc3RpbWF0ZSwgc3RkLmVycm9yKSAlPiUgCiAgYXJyYW5nZShBUlIpICU+JSAKICBpbm5lcl9qb2luKGNvZWZfbiwgYnkgPSBjKCJzcF9uYW1lIiwgInNleCIpKQoKQVJSX3ZhbHMgJT4lIAogIHNlbGVjdCgiU3BlY2llcyIgPSBzcF9uYW1lLCAKICAgICAgICAgIkdyb3VwIiA9IHNleCwgCiAgICAgICAgICJOIiA9IHNhbXBsZV9uLAogICAgICAgICBBUlIsIAogICAgICAgICAiRXJyb3IiID0gc3RkLmVycm9yKSAlPiUgCiAga25pdHI6OmthYmxlKCkKCm1sZV9jb2VmcyA9IGNvZWZmaWNpZW50cyhmdWxsLm1vZGVsKSRgc3BfbmFtZTpzZXhgICU+JSAKICBtdXRhdGUoImdyb3VwIiA9IHJvd25hbWVzKC4pKSAlPiUgCiAgc2VsZWN0KGdyb3VwLCAiaW50ZXJjZXB0IiA9ICIoSW50ZXJjZXB0KSIsICJBUlIiID0gdGVtcF9jZW50LCBzaXplX2NlbnQpICU+JSAKICBzZXBhcmF0ZShncm91cCwgaW50byA9IGMoInNwZWNpZXMiLCAic2V4IiksIHNlcCA9ICI6IiwgcmVtb3ZlID0gVFJVRSkgJT4lIAogIHJlbW92ZV9yb3duYW1lcygpCgptbGVfQVJSID0gbWxlX2NvZWZzICU+JSAgCiAgc2VsZWN0KHNwX25hbWUgPSBzcGVjaWVzLCBzZXgsIEFSUikgJT4lIAogIG11dGF0ZSgibW9kZWwiID0gIm1peGVkIGVmZmVjdHMiKSAgJT4lIAogIGlubmVyX2pvaW4oY29lZl9uLCBieSA9IGMoInNwX25hbWUiLCAic2V4IikpCgpBUlJfY29tcCA9IGJpbmRfcm93cyhtbGVfQVJSLCAKICAgICAgICAgICAgICAgICAgICAgbXV0YXRlKEFSUl92YWxzLCAibW9kZWwiID0gImxpbmVhciIpKQoKQVJSX2NvbXAgJT4lIAogIGdncGxvdChhZXMoeCA9IG1vZGVsLCB5ID0gQVJSLCBncm91cCA9IHNwX25hbWUpKSArIAogIGZhY2V0X2dyaWQoc3BfbmFtZX5zZXgpICsKICBnZW9tX3BvaW50KHNpemUgPSAzKSArIAogIGdlb21fbGluZShsaW5ld2lkdGggPSAxLjUpICsgCiAgc2NhbGVfeV9jb250aW51b3VzKGJyZWFrcyA9IGMoMCwgMC41LCAxKSkgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpICsgCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSAzMDAsIGhqdXN0ID0gMCwgdmp1c3QgPSAwLjUpKQoKYGBgCgpgYGB7ciBjb2VmLXN1bW1hcnksIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD01LCBpbmNsdWRlID0gRn0KY29lZl9wbG90ID0gbWxlX0FSUiAlPiUgCiAgbXV0YXRlKCJhYmJyIiA9IGNhc2Vfd2hlbigKICAgIHNwX25hbWUgPT0gIkVwaXNjaHVyYSBsYWN1c3RyaXMiIH4gIkUuIGxhYyIsCiAgICBzcF9uYW1lID09ICJMZXB0b2RpYXB0b211cyBtaW51dHVzIiB+ICJMLiBtaW4iLCAKICAgIHNwX25hbWUgPT0gIlNraXN0b2RpYXB0b211cyBvcmVnb25lbnNpcyIgfiAiUy4gb3JlIiwKICAgIHNwX25hbWUgPT0gIkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMiIH4gIkwuIHNpYyIsCiAgICBzcF9uYW1lID09ICJMaW1ub2NhbGFudXMgbWFjcnVydXMiIH4gIkwuIG1hYyIKICApKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gYWJiciwgeSA9IEFSUiwgY29sb3VyID0gc3BfbmFtZSwgc2hhcGUgPSBzZXgpKSArIAogICMgZ2VvbV9lcnJvcmJhcihhZXMoeW1pbiA9IEFSUiAtIHN0ZC5lcnJvciwgeW1heCA9IEFSUiArIHN0ZC5lcnJvciksCiAgIyAgICAgICAgICAgICAgIHdpZHRoID0gMC4zNSwgbGluZXdpZHRoID0gMSwgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDAuNSkpICsgCiAgZ2VvbV9wb2ludChzaXplID0gNSwKICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25fZG9kZ2Uod2lkdGggPSAwLjUpKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIGd1aWRlcyhjb2xvdXIgPSAibm9uZSIpICsgCiAgbGFicyh4ID0gIlNwZWNpZXMiLCAKICAgICAgIHkgPSAiQVJSIiwgCiAgICAgICBzaGFwZSA9ICJHcm91cCIpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCmNvZWZfbGltX3Bsb3QgPSBnZ3Bsb3QobWxlX0FSUiwgYWVzKHggPSBtZWFuX2N0bWF4LCB5ID0gQVJSKSkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBjb2xvdXIgPSAiZ3JleTcwIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyKSArIAogICMgZ2VvbV9lcnJvcmJhcihhZXMoY29sb3VyID0gc3BfbmFtZSwgeW1pbiA9IEFSUiAtIHN0ZC5lcnJvciwgeW1heCA9IEFSUiArIHN0ZC5lcnJvciksCiAgIyAgICAgICAgICAgICAgIHdpZHRoID0gMC41LCBsaW5ld2lkdGggPSAxLCBwb3NpdGlvbiA9IHBvc2l0aW9uX2RvZGdlKHdpZHRoID0gMC41KSkgKyAKICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBzcF9uYW1lLCBzaGFwZSA9IHNleCApLAogICAgICAgICAgICAgc2l6ZSA9IDUsCiAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX2RvZGdlKHdpZHRoID0gMC41KSkgKwogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIGd1aWRlcyhjb2xvdXIgPSAibm9uZSIpICsgCiAgbGFicyh4ID0gIk1lYW4gQ1RtYXggKMKwQykiLCAKICAgICAgIHkgPSAiQVJSIiwgCiAgICAgICBzaGFwZSA9ICJHcm91cCIpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCmdnYXJyYW5nZShjb2VmX3Bsb3QsIGNvZWZfbGltX3Bsb3QsIGNvbW1vbi5sZWdlbmQgPSBULCBsZWdlbmQgPSAicmlnaHQiKQpgYGAKCiMjIyBTZXggYW5kIHN0YWdlIHZhcmlhdGlvbiBpbiB0aGVybWFsIGxpbWl0cyAKUHJldmlvdXMgc2VjdGlvbnMgaGF2ZSBnZW5lcmFsbHkgbHVtcGVkIGp1dmVuaWxlLCBmZW1hbGUsIGFuZCBtYWxlIGluZGl2aWR1YWxzIHRvZ2V0aGVyLiBUaGVyZSBtYXkgYmUgaW1wb3J0YW50IHN0YWdlLSBvciBzZXgtc3BlY2lmaWMgZGlmZmVyZW5jZXMgaW4gQ1RtYXggdGhvdWdoLiBGb3Igc2V2ZXJhbCBzcGVjaWVzLCB3ZSBoYXZlIG1lYXN1cmVtZW50cyBmb3IgaW5kaXZpZHVhbHMgaW4gZGlmZmVyZW50IHN0YWdlcyBvciBvZiBkaWZmZXJlbnQgc2V4ZXMuIAoKYGBge3Igc2V4LXN0YWdlLXRhYmxlfQpzZXhfc2FtcGxlX3NpemVzID0gY3RtYXhfcmVzaWRzICU+JSAgCiAgZ3JvdXBfYnkoc3BfbmFtZSwgc2V4KSAlPiUgIAogIHN1bW1hcmlzZShudW0gPSBuKCkpICU+JSAgCiAgcGl2b3Rfd2lkZXIoaWRfY29scyA9IHNwX25hbWUsCiAgICAgICAgICAgICAgbmFtZXNfZnJvbSA9IHNleCwgCiAgICAgICAgICAgICAgdmFsdWVzX2Zyb20gPSBudW0sCiAgICAgICAgICAgICAgdmFsdWVzX2ZpbGwgPSAwKSAlPiUgCiAgc2VsZWN0KCJTcGVjaWVzIiA9IHNwX25hbWUsICJKdXZlbmlsZSIgPSBqdXZlbmlsZSwgIkZlbWFsZSIgPSBmZW1hbGUsICJNYWxlIiA9IG1hbGUpCgprbml0cjo6a2FibGUoc2V4X3NhbXBsZV9zaXplcywgYWxpZ24gPSAiYyIpCmBgYAoKVGhlIGZlbWFsZS1tYWxlIGFuZCBmZW1hbGUtanV2ZW5pbGUgY29tcGFyaXNvbnMgc2hvdyB0aGF0IHRoZXJlIGFyZSBnZW5lcmFsbHkgbm8gZGlmZmVyZW5jZXMgaW4gdGhlcm1hbCBsaW1pdHMgYmV0d2VlbiB0aGVzZSBncm91cHMuIAoKYGBge3IgY3RtYXgtc2V4LCBmaWcud2lkdGg9NywgZmlnLmhlaWdodD03fQpjdG1heF9yZXNpZHMgJT4lIAogIGZpbHRlcihzcF9uYW1lICVpbiUgZmlsdGVyKHNleF9zYW1wbGVfc2l6ZXMsIE1hbGUgPiAwLCBGZW1hbGUgPiAwKSRTcGVjaWVzICYgCiAgICAgICAgICAgc2V4ICE9ICJqdXZlbmlsZSIpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBzZXgsIHkgPSByZXNpZHMsIGNvbG91ciA9IHNwX25hbWUsIGdyb3VwID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfd3JhcChzcF9uYW1lfi4sIG5jb2wgPSAyKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgbGluZXdpZHRoID0gMSkgKyAKICBnZW9tX3BvaW50KHNpemUgPSAzLAogICAgICAgICAgICAgYWxwaGEgPSAwLjUsCiAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX2ppdHRlcihoZWlnaHQgPSAwLCB3aWR0aCA9IDAuMDUpKSArICAKICBsYWJzKHggPSAiU2V4IiwgCiAgICAgICB5ID0gIkNUbWF4IFJlc2lkdWFscyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfYncoYmFzZV9zaXplID0gMTgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiLCAKICAgICAgICBwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpKQpgYGAKCmBgYHtyIGN0bWF4LXN0YWdlLCBmaWcud2lkdGg9NywgZmlnLmhlaWdodD03fQpjdG1heF9yZXNpZHMgJT4lIAogIGZpbHRlcihzcF9uYW1lICVpbiUgZmlsdGVyKHNleF9zYW1wbGVfc2l6ZXMsIEp1dmVuaWxlID4gMCAmIEZlbWFsZSA+IDApJFNwZWNpZXMgJiAKICAgICAgICAgICBzZXggIT0gIm1hbGUiKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gc2V4LCB5ID0gcmVzaWRzLCBjb2xvdXIgPSBzcF9uYW1lLCBncm91cCA9IHNwX25hbWUpKSArIAogIGZhY2V0X3dyYXAoc3BfbmFtZX4uLCBuY29sID0gMikgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9wb2ludChzaXplID0gMywKICAgICAgICAgICAgIGFscGhhID0gMC41LAogICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9qaXR0ZXIoaGVpZ2h0ID0gMCwgd2lkdGggPSAwLjA1KSkgKyAgCiAgbGFicyh4ID0gIlNleCIsIAogICAgICAgeSA9ICJDVG1heCAowrBDKSIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfYncoYmFzZV9zaXplID0gMTgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiLCAKICAgICAgICBwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpKQpgYGAKCmBgYHtyIHRyYWl0LXZhcmlhbmNlLWNvbGwtdGVtcCwgaW5jbHVkZSA9IEZ9CiMgCiMgR2l2ZW4gdGhlIGxvbmcgZ2VuZXJhdGlvbiB0aW1lcyBvZiB0aGVzZSBjb3BlcG9kcywgZGVjcmVhc2VzIGluIHRyYWl0IHZhcmlhbmNlIG1heSBpbmRpY2F0ZSBzZWxlY3Rpb24gb3ZlciB0aGUgc2Vhc29uYWwgY3ljbGUuIFNob3duIGJlbG93IGFyZSB0aGUgdmFyaWFuY2UgaW4gb2JzZXJ2ZWQgQ1RtYXggYW5kIHNpemUsIHBsb3R0ZWQgYWdhaW5zdCBjb2xsZWN0aW9uIGRhdGUuIFZhcmlhbmNlIGRlY3JlYXNlcyBpbiAqU2tpc3RvZGlhcHRvbXVzKiwgYnV0IHRoaXMgcGF0dGVybiBpcyBkcml2ZW4gYnkgYSBzaW5nbGUgY29sbGVjdGlvbiB3aXRoIGhpZ2ggdmFyaWFuY2UgZWFybHkgaW4gdGhlIHllYXIuIFNpemUgdmFyaWFuY2UgaW5jcmVhc2VzIHNsaWdodGx5IGluICpTa2lzdG9kaWFwdG9tdXMqLiBWYXJpYW5jZSBpbiBib3RoIENUbWF4IGFuZCBzaXplIGlzIGZhaXJseSBjb25zdGFudCBpbiAqTGVwdG9kaWFwdG9tdXMgbWludXR1cyosIHRoZSBvbmx5IG90aGVyIHNwZWNpZXMgY29sbGVjdGVkIGFjcm9zcyB0aGUgZW50aXJlIHNldCBvZiBzYW1wbGVzIHRodXMgZmFyLiAKIyAKIyBnZ3Bsb3QoZHJvcF9uYShhZHVsdF9zdW1tYXJpZXMsIGN0bWF4X3ZhciksIGFlcyh4ID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpLCB5ID0gY3RtYXhfdmFyLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKIyAgIGZhY2V0X3dyYXAoc3BfbmFtZX4uLCBzY2FsZXMgPSAiZnJlZV95IikgKyAKIyAgIGdlb21fcG9pbnQoc2l6ZSA9IDIpICsgCiMgICBnZW9tX3Ntb290aChzZSA9IEYpICsgCiMgICBsYWJzKHggPSAiQ29sbGVjdGlvbiBUZW1wLiAowrBDKSIsIAojICAgICAgICB5ID0gIkNUbWF4IFZhcmlhbmNlIikgKyAKIyAgIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAojICAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAojICAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQojIAojIGdncGxvdChkcm9wX25hKGFkdWx0X3N1bW1hcmllcywgc2l6ZV92YXIpLCBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwgeSA9IHNpemVfdmFyLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKIyAgIGZhY2V0X3dyYXAoc3BfbmFtZX4uKSArIAojICAgZ2VvbV9wb2ludChzaXplID0gMikgKyAKIyAgIGdlb21fc21vb3RoKHNlID0gRikgKyAKIyAgIGxhYnMoeCA9ICJDb2xsZWN0aW9uIFRlbXAuICjCsEMpIiwgCiMgICAgICAgIHkgPSAiU2l6ZSBWYXJpYW5jZSIpICsgCiMgICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKIyAgIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKIyAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgoKIyMjIFRyYWl0IENvcnJlbGF0aW9ucyBhbmQgVHJhZGUtb2ZmcwoKQSByZWxhdGlvbnNoaXAgYmV0d2VlbiBzaXplIGFuZCB1cHBlciB0aGVybWFsIGxpbWl0cyBoYXMgYmVlbiBzdWdnZXN0ZWQgaW4gYSB3aWRlIHJhbmdlIG9mIG90aGVyIHRheGEuIFNob3duIGJlbG93IGFyZSB0aGUgbWVhc3VyZWQgdXBwZXIgdGhlcm1hbCBsaW1pdHMgcGxvdHRlZCBhZ2FpbnN0IHByb3NvbWUgbGVuZ3RoLiBUaGUgb3ZlcmFsbCByZWxhdGlvbnNoaXAgKGluY2x1c2l2ZSBvZiBhbGwgc3BlY2llcykgaXMgc2hvd24gYXMgdGhlIGJsYWNrIGxpbmUgaW4gdGhlIGJhY2tncm91bmQuIFJlZ3Jlc3Npb25zIGZvciBlYWNoIGluZGl2aWR1YWwgc3BlY2llcyBhcmUgYWxzbyBzaG93bi4gQWNyb3NzIHRoZSBlbnRpcmUgYXNzZW1ibGFnZSwgdGhlcmUgaXMgYSBzdHJvbmcgZGVjcmVhc2UgaW4gdGhlcm1hbCBsaW1pdHMgd2l0aCBpbmNyZWFzaW5nIHNpemUuICAKCmBgYHtyIGN0bWF4LXNpemUsIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD03fQoKZnVsbF9kYXRhICU+JSAKICAjZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lICAKICBnZ3Bsb3QoIGFlcyh4ID0gc2l6ZSwgeSA9IGN0bWF4LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBnZW9tX3Ntb290aChkYXRhID0gZnVsbF9kYXRhLCAKICAgICAgICAgICAgICBhZXMoeCA9IHNpemUsIHkgPSBjdG1heCksCiAgICAgICAgICAgICAgbWV0aG9kID0gImxtIiwgCiAgICAgICAgICAgICAgY29sb3VyID0iYmxhY2siLCAKICAgICAgICAgICAgICBsaW5ld2lkdGggPSAyLjUpICsgCiAgZ2VvbV9wb2ludChzaXplID0gMiwgYWxwaGEgPSAwLjMpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyKSArIAogIGxhYnMoeCA9ICJMZW5ndGggKG1tKSIsIAogICAgICAgeSA9ICJDVG1heCAowrBDKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCmBgYAoKU2hvd24gaGVyZSBpcyB0aGUgcmVsYXRpb25zaGlwIGZvciBlYWNoIHNwZWNpZXMgaW5kaXZpZHVhbGx5LiAKCmBgYHtyIGluZC1zcC1jdG1heC1zaXplLCBmaWcud2lkdGg9OSwgZmlnLmhlaWdodD02fQpmdWxsX2RhdGEgJT4lIAogICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JSBmaWx0ZXIobigpID4yKSAlPiUgZmlsdGVyKCFzdHJfZGV0ZWN0KHNwX25hbWUsIHBhdHRlcm4gPSAia2luZHRpIikpICU+JSAKICBnZ3Bsb3QoIGFlcyh4ID0gc2l6ZSwgeSA9IGN0bWF4LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBmYWNldF93cmFwKHNwX25hbWV+Liwgc2NhbGVzID0gImZyZWUiLCBucm93ID0gMikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuOCkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDIpICsgCiAgbGFicyh4ID0gIkxlbmd0aCAobW0pIiwgCiAgICAgICB5ID0gIkNUbWF4ICjCsEMpIiwKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCmBgYAoKU2hvd24gYmVsb3cgaXMgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIG1lYW4gc2l6ZSBhbmQgbWVhbiB0aGVybWFsIGxpbWl0cyBmb3IgZmVtYWxlcyBvZiBlYWNoIHNwZWNpZXMuIFdlIHNlZSB0aGF0IGxhcmdlciBzcGVjaWVzIHdpdGhpbiB0aGUgY29tbXVuaXR5IHRlbmQgdG8gaGF2ZSBhIGxvd2VyIHRoZXJtYWwgbGltaXQgdGhhbiBzbWFsbGVyIHNwZWNpZXMuIAoKYGBge3IgbWVhbi1jdG1heC1tZWFuLXNpemUtcGxvdCwgZmlnLndpZHRoPTksIGZpZy5oZWlnaHQ9NX0KZnVsbF9kYXRhICU+JSAKICBncm91cF9ieShzcF9uYW1lLCBzZXgpICU+JSAKICBzdW1tYXJpemUobWVhbl9jdG1heCA9IG1lYW4oY3RtYXgsIG5hLnJtID0gVCksCiAgICAgICAgICAgIG1lYW5fc2l6ZSA9IG1lYW4oc2l6ZSwgbmEucm0gPSBUKSkgJT4lIAogICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gbWVhbl9zaXplLCB5ID0gbWVhbl9jdG1heCkpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyLCBjb2xvdXIgPSAiYmxhY2siKSArIAogIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IHNwX25hbWUsIHNoYXBlID0gc2V4KSwKICAgICAgICAgICAgIHNpemUgPSA1KSArIAogIGxhYnMoeCA9ICJMZW5ndGggKG1tKSIsIAogICAgICAgeSA9ICJDVG1heCAowrBDKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgpgYGB7ciBjdG1heHJlc2lkcy1zaXplLCBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9NywgaW5jbHVkZSA9IEZ9CmN0bWF4X3Jlc2lkcyAlPiUgCiAgI2ZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBzaXplX3Jlc2lkcywgeSA9IHJlc2lkcywgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfd3JhcChzcF9uYW1lfi4pICsgCiAgZ2VvbV9wb2ludChzaXplID0gMikgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDIpICsgCiAgbGFicyh4ID0gIkxlbmd0aCAobW0pIiwgCiAgICAgICB5ID0gIkNUbWF4ICjCsEMpIikgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCmBgYAoKU2hvd24gaGVyZSBpcyB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gZmVjdW5kaXR5IGFuZCBzaXplLCBzaG93aW5nIHRoZSBjbGFzc2ljIHBhdHRlcm4gb2YgaW5jcmVhc2luZyBlZ2cgcHJvZHVjdGlvbiB3aXRoIGluY3JlYXNpbmcgc2l6ZS4gCgpgYGB7ciBmZWN1bmRpdHktc2l6ZSwgZmlnLndpZHRoPTEwLCBmaWcuaGVpZ2h0PTd9CmN0bWF4X3Jlc2lkcyAlPiUgIAogIGRyb3BfbmEoZmVjdW5kaXR5KSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gc2l6ZSwgeSA9IGZlY3VuZGl0eSwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyKSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDIsIGFscGhhID0gMC41KSArIAogIGxhYnMoeCA9ICJQcm9zb21lIGxlbmd0aCAobW0pIiwgCiAgICAgICB5ID0gIkZlY3VuZGl0eSAoIyBFZ2dzKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgpJbmRpdmlkdWFscyBtYXkgYWxzbyBhbGxvY2F0ZSBlbmVyZ3kgdG8gZGlmZmVyZW50IGZpdG5lc3MgcmVsYXRlZCB0cmFpdHMsIHByaW9yaXRpemluZyByZXByb2R1Y3RpdmUgb3V0cHV0IG92ZXIgZW52aXJvbm1lbnRhbCB0b2xlcmFuY2UsIGZvciBleGFtcGxlLiBTaG93biBiZWxvdyBpcyB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gQ1RtYXggcmVzaWR1YWxzIChhZ2FpbiwgY29udHJvbGxpbmcgZm9yIHRoZSBlZmZlY3RzIG9mIGNvbGxlY3Rpb24gdGVtcGVyYXR1cmUpIGFnYWluc3QgZmVjdW5kaXR5LiBXZSBjYW4gc2VlIGNsZWFybHkgdGhhdCBpbmRpdmlkdWFscyB3aXRoIGluY3JlYXNlZCBmZWN1bmRpdHkgYXJlIG5vdCBkZWNyZWFzaW5nIHRoZXJtYWwgbGltaXRzLCBzdWdnZXN0aW5nIHRoYXQgdGhlcmUgaXMgbm8gZW5lcmdldGljIHRyYWRlLW9mZiBiZXR3ZWVuIHRoZXNlIHRyYWl0cy4gCgpgYGB7ciwgY3RtYXgtZmVjdW5kaXR5LCBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9N30KY3RtYXhfcmVzaWRzICU+JSAgCiAgZHJvcF9uYShmZWN1bmRpdHkpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSByZXNpZHMsIHkgPSBmZWN1bmRpdHksIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgbGluZXdpZHRoID0gMikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuNSkgKyAKICBsYWJzKHggPSAiQ1RtYXggUmVzaWR1YWxzIiwgCiAgICAgICB5ID0gIkZlY3VuZGl0eSAoIyBFZ2dzKSIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCmZpdG5lc3MubW9kZWwgPSBsbShkYXRhID0gY3RtYXhfcmVzaWRzLCAKICAgZmVjdW5kaXR5IH4gcmVzaWRzICogc3BfbmFtZSkKCmNhcjo6QW5vdmEoZml0bmVzcy5tb2RlbCkKCmVtbWVhbnM6OmVtdHJlbmRzKGZpdG5lc3MubW9kZWwsICB2YXIgPSAicmVzaWRzIiwic3BfbmFtZSIpCmBgYAoKIyMgT3RoZXIgcGF0dGVybnMgaW4gdmFyaWF0aW9uIAoKKkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMqIGlzIHRoZSBtb3N0IGFidW5kYW50IHNwZWNpZXMgZHVyaW5nIHRoZSB3aW50ZXIuIFRoZXJlIHdhcyBhIGxhcmdlIHNoaWZ0IGluIHRoZSBzaXplIG9mIG1hdHVyZSBmZW1hbGVzIHRvd2FyZHMgdGhlIGVuZCBvZiBEZWNlbWJlci4gVGhlc2UgbGFyZ2UgYW5kIHNtYWxsIGluZGl2aWR1YWxzIGFyZSB0aGUgc2FtZSBzcGVjaWVzIChjb25maXJtZWQgdmlhIENPSSBzZXF1ZW5jaW5nKSwgc3VnZ2VzdGluZyB0aGlzIHNoaWZ0IG1heSByZWZsZWN0IGEgdHJhbnNpdGlvbiBmcm9tIG9uZSBnZW5lcmF0aW9uIHRvIGFub3RoZXIgYW5kIHRoYXQsIHVubGlrZSBpbiBtYW55IG90aGVyIGxha2VzLCB0aGVyZSBhcmUgdHdvIGdlbmVyYXRpb25zIG9mICpMLiBzaWNpbGlzKiBwZXIgeWVhciBpbiBMYWtlIENoYW1wbGFpbi4gVGhpcyBzaXplIGRpZmZlcmVuY2UgbWF5IGJlIGNhdXNlZCBieSBkaWZmZXJlbmNlcyBpbiB0aGUgZGV2ZWxvcG1lbnRhbCBlbnZpcm9ubWVudHMuIEZvciBleGFtcGxlLCBpbmRpdmlkdWFscyBkZXZlbG9waW5nIGluIEphbnVhcnkgZ3JvdyB1cCBhdCB2ZXJ5IGxvdyB0ZW1wZXJhdHVyZXMsIGFuZCB0aGVyZWZvcmUgbWF5IHJlYWNoIGxhcmdlciBzaXplcy4gVGhlc2UgaW5kaXZpZHVhbHMgb3ZlcnN1bW1lciBpbiBkZWVwIHdhdGVycywgdGhlbiByZS1lbWVyZ2UgaW4gT2N0b2JlciBhbmQgcHJvZHVjZSBhIG5ldyBnZW5lcmF0aW9uLiBXYXRlciB0ZW1wZXJhdHVyZXMgYXJlIHN0aWxsIGZhaXJseSBoaWdoIHRocm91Z2ggTm92ZW1iZXIsIHdoaWNoIHJlc3VsdHMgaW4gYSBnZW5lcmF0aW9uIG9mIHNtYWxsZXIgaW5kaXZpZHVhbHMsIHdoaWNoIG1hdHVyZSBpbiB0aW1lIHRvIHByb2R1Y2UgYSBuZXcgZ2VuZXJhdGlvbiBpbiBKYW51YXJ5LiAKClNob3duIGJlbG93IGlzIHRoZSBkaXN0cmlidXRpb24gb2YgcGFpcndpc2UgZGlzdGFuY2VzIGJldHdlZW4gQ09JIHNlcXVlbmNlcyBvZiBsYXJnZSBhbmQgc21hbGwgbW9ycGhzLiBEaXN0YW5jZXMgaW4gYm90aCB3aXRoaW4tIGFuZCBhY3Jvc3MtbW9ycGggY29tcGFyaXNvbnMgYXJlIHNtYWxsLiAKCmBgYHtyfQppbmRfZGlzdCA9IGFwZTo6ZGlzdC5kbmEoc2ljX2RuYWJpbiwgbW9kZWwgPSAicmF3IikgJT4lIGFzLm1hdHJpeCAlPiUgCiAgYXNfdGliYmxlKCkgJT4lCiAgbXV0YXRlKCJpbmQxIiA9IGNvbG5hbWVzKC4pKSAlPiUgCiAgcGl2b3RfbG9uZ2VyKC1pbmQxLCBuYW1lc190byA9ICJpbmQyIiwgdmFsdWVzX3RvID0gImRpc3QiKSAlPiUKICBtdXRhdGUoaW5kMSA9IGZhY3RvcihpbmQxKSwKICAgICAgICAgaW5kMiA9IGZhY3RvcihpbmQyKSkgJT4lIAogIGZpbHRlcighKGluZDEgPT0gInNvcmUxIiB8IGluZDIgPT0gInNvcmUxIikpICU+JSAKICBtdXRhdGUoCiAgICBpbmQxID0gY2FzZV93aGVuKAogICAgICBpbmQxID09ICJTMSIgfiAic21hbGwxIiwKICAgICAgaW5kMSA9PSAiUzMiIH4gInNtYWxsMyIsCiAgICAgIGluZDEgPT0gImxzaWMzIiB+ICJzbWFsbDQiLAogICAgICBpbmQxID09ICJsc2ljNSIgfiAic21hbGw2IiwKICAgICAgaW5kMSA9PSAibHNpYzkiIH4gInNtYWxsOCIsCiAgICAgIGluZDEgPT0gImxzaWMxMCIgfiAic21hbGw5IiwKICAgICAgaW5kMSA9PSAibHNpYzExIiB+ICJzbWFsbDEwIiwKICAgICAgaW5kMSA9PSAiTDEiIH4gImxhcmdlMSIsCiAgICAgIGluZDEgPT0gIkwyIiB+ICJsYXJnZTIiLAogICAgICBpbmQxID09ICJMMyIgfiAibGFyZ2UzIiwKICAgICAgaW5kMSA9PSAibHNpYzEiIH4gImxhcmdlNCIsCiAgICAgIGluZDEgPT0gImxzaWMyIiB+ICJsYXJnZTUiLAogICAgICBpbmQxID09ICJsc2ljNyIgfiAibGFyZ2U2IiwKICAgICAgaW5kMSA9PSAibHNpYzgiIH4gImxhcmdlNyIpLAogICAgaW5kMiA9IGNhc2Vfd2hlbigKICAgICAgaW5kMiA9PSAiUzEiIH4gInNtYWxsMSIsCiAgICAgIGluZDIgPT0gIlMzIiB+ICJzbWFsbDMiLAogICAgICBpbmQyID09ICJsc2ljMyIgfiAic21hbGw0IiwKICAgICAgaW5kMiA9PSAibHNpYzUiIH4gInNtYWxsNiIsCiAgICAgIGluZDIgPT0gImxzaWM5IiB+ICJzbWFsbDgiLAogICAgICBpbmQyID09ICJsc2ljMTAiIH4gInNtYWxsOSIsCiAgICAgIGluZDIgPT0gImxzaWMxMSIgfiAic21hbGwxMCIsCiAgICAgIGluZDIgPT0gIkwxIiB+ICJsYXJnZTEiLAogICAgICBpbmQyID09ICJMMiIgfiAibGFyZ2UyIiwKICAgICAgaW5kMiA9PSAiTDMiIH4gImxhcmdlMyIsCiAgICAgIGluZDIgPT0gImxzaWMxIiB+ICJsYXJnZTQiLAogICAgICBpbmQyID09ICJsc2ljMiIgfiAibGFyZ2U1IiwKICAgICAgaW5kMiA9PSAibHNpYzciIH4gImxhcmdlNiIsCiAgICAgIGluZDIgPT0gImxzaWM4IiB+ICJsYXJnZTciKSwKICAgICdjb21wYXJpc29uJyA9IGNhc2Vfd2hlbigKICAgICAgc3RyX2RldGVjdChpbmQxLCBwYXR0ZXJuID0gImxhcmdlIikgJiBzdHJfZGV0ZWN0KGluZDIsIHBhdHRlcm4gPSAibGFyZ2UiKSB+ICJ3aXRoaW4iLAogICAgICBzdHJfZGV0ZWN0KGluZDEsIHBhdHRlcm4gPSAic21hbGwiKSAmIHN0cl9kZXRlY3QoaW5kMiwgcGF0dGVybiA9ICJzbWFsbCIpIH4gIndpdGhpbiIsIAogICAgICBzdHJfZGV0ZWN0KGluZDEsIHBhdHRlcm4gPSAibGFyZ2UiKSAmIHN0cl9kZXRlY3QoaW5kMiwgcGF0dGVybiA9ICJzbWFsbCIpIH4gImFjcm9zcyIsCiAgICAgIHN0cl9kZXRlY3QoaW5kMSwgcGF0dGVybiA9ICJzbWFsbCIpICYgc3RyX2RldGVjdChpbmQyLCBwYXR0ZXJuID0gImxhcmdlIikgfiAiYWNyb3NzIgogICAgKSkgCgpnZ3Bsb3QoaW5kX2Rpc3QsIGFlcyhkaXN0LCBmaWxsID0gY29tcGFyaXNvbikpICsKICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aCA9IDAuMDA1KSArIAogIGxhYnMoeCA9ICJEaXN0YW5jZSIpICsgCiAgdGhlbWVfbWF0dCgpCmBgYAoKCmBgYHtyfQpmdWxsX2RhdGEgJT4lICAKICBmaWx0ZXIoc3BfbmFtZSA9PSAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyIpICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoY29sbGVjdGlvbl9kYXRlKSAlPiUgCiAgbXV0YXRlKHNpemVfY2VudGVyID0gc2NhbGUoc2l6ZSwgY2VudGVyID0gVCwgc2NhbGUgPSBGKSkgJT4lIAogIGdncGxvdChhZXMoeSA9IGNvbGxlY3Rpb25fZGF0ZSwgeCA9IHNpemUsIGZpbGwgPSBjb2xsZWN0aW9uX3RlbXApKSArIAogIGdlb21fZGVuc2l0eV9yaWRnZXMoYmFuZHdpZHRoID0gMC4wNCkgKyAKICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSAwLjg5KSArIAogIGxhYnMoeCA9ICJTaXplIChtbSkiLAogICAgICAgeSA9ICJEYXRlIiwgCiAgICAgICBmaWxsID0gIkNvbGwuIFRlbXAuICjCsEMpIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiLAogICAgICAgIGF4aXMudGV4dC55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpCmBgYAoKYGBge3J9CmZ1bGxfZGF0YSAlPiUgIAogIGZpbHRlcihzcF9uYW1lID09ICJMZXB0b2RpYXB0b211cyBtaW51dHVzIikgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAKICBncm91cF9ieShjb2xsZWN0aW9uX2RhdGUpICU+JSAKICBtdXRhdGUoc2l6ZV9jZW50ZXIgPSBzY2FsZShzaXplLCBjZW50ZXIgPSBULCBzY2FsZSA9IEYpKSAlPiUgCiAgZ2dwbG90KGFlcyh5ID0gY29sbGVjdGlvbl9kYXRlLCB4ID0gc2l6ZSwgZmlsbCA9IGNvbGxlY3Rpb25fdGVtcCkpICsgCiAgZ2VvbV9kZW5zaXR5X3JpZGdlcyhiYW5kd2lkdGggPSAwLjA0KSArIAogIGdlb21fdmxpbmUoeGludGVyY2VwdCA9IDAuNjkpICsgCiAgbGFicyh4ID0gIlNpemUgKG1tKSIsCiAgICAgICB5ID0gIkRhdGUiLCAKICAgICAgIGZpbGwgPSAiQ29sbC4gVGVtcC4gKMKwQykiKSArIAogIGNvb3JkX2NhcnRlc2lhbih4bGltID0gYygwLjUsMC45KSkgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiLAogICAgICAgIGF4aXMudGV4dC55ID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMikpCmBgYAoKYGBge3J9CmlmKHByZWRpY3RfdnVsbiA9PSBGKXsKICBrbml0cjo6a25pdF9leGl0KCkKfQpgYGAKCgoKIyMgUHJlZGljdGluZyBWdWxuZXJhYmlsaXR5IApVc2luZyB0aGUgb2JzZXJ2ZWQgdGhlcm1hbCBsaW1pdCBkYXRhLCB3ZSBjYW4gcHJvZHVjZSBhIGhpbmRjYXN0IG9mIHRoZXJtYWwgc3RyZXNzIGZvciBMYWtlIENoYW1wbGFpbiBjb3BlcG9kcy4gRm9yIHRoZXNlIGluaXRpYWwgYXNzYXlzLCB3ZSB3aWxsIGRlZmluZSB0aGVybWFsIHN0cmVzcyBhcyBhbnkgdGltZSB3aGVuIG1heGltdW0gZGFpbHkgd2F0ZXIgdGVtcGVyYXR1cmUgaXMgd2l0aGluIDLCsEMgb2YgY29wZXBvZCBDVG1heCBvciBoaWdoZXIuIFdlIHdpbGwgdXNlIHRocmVlIGRpZmZlcmVudCBzY2VuYXJpb3M6IDEpIHRoZSBhdmVyYWdlIENUbWF4IGZvciBlYWNoIHNwZWNpZXMsIDIpIENUbWF4IHByZWRpY3RlZCB1c2luZyBjb2xsZWN0aW9uIHRlbXBlcmF0dXJlcywgYW5kIDMpIGZvciBzcGVjaWVzIHRoYXQgaGF2ZSBzdWZmaWNpZW50IGRhdGEsIENUbWF4IHByZWRpY3RlZCB1c2luZyB3aGljaGV2ZXIgZW52aXJvbm1lbnRhbCBmYWN0b3IgaXMgdGhlIHN0cm9uZ2VzdCBjYW5kaWRhdGUgZm9yIGRyaXZpbmcgYWNjbGltYXRpb24uIEluIGFsbCBjYXNlcywgZGF0YSBpcyBmaWx0ZXJlZCB0byBqdXN0IHRoZXJtYWwgbGltaXRzIG9mIGFkdWx0IGZlbWFsZXMuIAoKIyMjIFNjZW5hcmlvIDEKYGBge3J9Cm1lYW5fY3RtYXggPSBmdWxsX2RhdGEgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lIAogIHN1bW1hcml6ZSgibWVhbl9jdG1heCIgPSBtZWFuKGN0bWF4KSkgJT4lIAogIGFycmFuZ2UobWVhbl9jdG1heCkKCmtuaXRyOjprYWJsZShtZWFuX2N0bWF4KQpgYGAKCmBgYHtyfQojICMgQ29uc3RydWN0cyB0aGUgVVJMIGZvciB0aGUgZnVsbCB0ZW1wZXJhdHVyZSBkYXRhIHNldDsgUlVOIFRISVMgT05DRQojIGhpbmRfdXJsID0gY29uc3RydWN0TldJU1VSTChzaXRlTnVtYmVycyA9IHNpdGVOdW1iZXIsIHBhcmFtZXRlckNkID0gcGFyYW1ldGVyQ2QsIHNlcnZpY2UgPSAidXYiKQojIAojIGhpbmRfdGVtcF9kYXRhID0gaW1wb3J0V2F0ZXJNTDEoaGluZF91cmwsIGFzRGF0ZVRpbWUgPSBUKSAlPiUKIyAgIG11dGF0ZSgiZGF0ZSIgPSBhcy5EYXRlKGRhdGVUaW1lKSkgJT4lCiMgICBzZWxlY3QoZGF0ZSwgInRlbXAiID0gWF8wMDAxMF8wMDAwMCkKIyAKIyB3cml0ZS50YWJsZSh4ID0gaGluZF90ZW1wX2RhdGEsIGZpbGUgPSAiaGluZGNhc3RfdGVtcHMuY3N2Iiwgcm93Lm5hbWVzID0gRiwgc2VwID0gIiwiKQpgYGAKCmBgYHtyfQojIGdncGxvdChoaW5kX3RlbXBfZGF0YSwgYWVzKHggPSBkYXRlLCB5ID0gdGVtcCkpICsgCiMgICBnZW9tX2xpbmUobGluZXdpZHRoID0gMC4xKSArIAojICAgbGFicyh4ID0gIkRhdGUiLCAKIyAgICAgICAgeSA9ICJXYXRlciBUZW1wZXJhdHVyZSAowrBDKSIpICsKIyAgIHRoZW1lX21hdHQoKQpgYGAKCkluIHRoZSBzaW1wbGVzdCBzY2VuYXJpbywgc3BlY2llcyB0aGVybWFsIGxpbWl0cyBhcmUgc3RhdGljIHRocm91Z2ggdGltZSwgcmVwcmVzZW50ZWQgYnkgdGhlIGF2ZXJhZ2UgQ1RtYXggb2YgYWR1bHQgZmVtYWxlIGNvcGVwb2RzLiBJbiB0aGlzIHNjZW5hcmlvLCBvbmx5IHRocmVlIG9mIHRoZSBzZXZlbiBvYnNlcnZlZCBzcGVjaWVzIGFyZSBleHBvc2VkIHRvIHRoZXJtYWwgc3RyZXNzICh0ZW1wZXJhdHVyZXMgd2l0aGluIDXCsEMgb2YgQ1RtYXgpLiBUZW1wZXJhdHVyZXMgYXBwcm9hY2hlZCB0aGUgdGhlcm1hbCBsaW1pdCBvZiAqTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyogb24gYSBoYW5kZnVsIG9mIGRheXMuIEJ5IGNvbnRyYXN0LCAqU2VuZWNlbGxhIGNhbGFub2lkZXMqIGFuZCAqTGltbm9jYWxhbnVzIG1hY3J1cnVzKiB3ZXJlIGJvdGggZXhwb3NlZCB0byBzdWJzdGFudGlhbCB0aGVybWFsIHN0cmVzcyB0aHJvdWdob3V0IGEgbGFyZ2UgcG9ydGlvbiBvZiB0aGUgeWVhciwgbGlrZWx5IGV4cGxhaW5pbmcgd2h5IHRoZXNlIHNwZWNpZXMgYXJlIGFic2VudCBmcm9tIHRoZSBjb21tdW5pdHkgZm9yIHRoZSBzdW1tZXIgYW5kIGZhbGwgcGVyaW9kcy4gCgpgYGB7ciBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9NX0KaGluZDFfZGF0YSA9IGhpbmRfdGVtcF9kYXRhICU+JSAKICBncm91cF9ieShkYXRlKSAlPiUgCiAgc3VtbWFyaXplKCJkYWlseV9tYXgiID0gbWF4KHRlbXApLAogICAgICAgICAgICAiZGFpbHlfbWVhbiIgPSBtZWFuKHRlbXApLCkgJT4lIAogIGJpbmRfY29scyhwaXZvdF93aWRlcihtZWFuX2N0bWF4LCBuYW1lc19mcm9tID0gc3BfbmFtZSwgdmFsdWVzX2Zyb20gPSBtZWFuX2N0bWF4KSkgJT4lICAKICBwaXZvdF9sb25nZXIoY29scyA9IGMoLWRhdGUsIC1kYWlseV9tYXgsIC1kYWlseV9tZWFuKSwKICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAic3BlY2llcyIsIAogICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAibWVhbl9jdG1heCIpICU+JSAgCiAgbXV0YXRlKGxpbV9kaWZmID0gbWVhbl9jdG1heCAtIGRhaWx5X21heCkgJT4lICAKICBtdXRhdGUoZG95ID0geWRheShkYXRlKSwKICAgICAgICAgIm1ldGhvZCIgPSAiTm9fYWNjbGltYXRpb24iKQoKaGluZF9kYWlseV90ZW1wX2RhdGEgPSBoaW5kX3RlbXBfZGF0YSAlPiUKICB1bmdyb3VwKCkgJT4lIAogIGdyb3VwX2J5KGRhdGUpICU+JSAKICBzdW1tYXJpc2UobWVhbl90ZW1wID0gbWVhbih0ZW1wKSwKICAgICAgICAgICAgbWVkX3RlbXAgPSBtZWRpYW4odGVtcCksCiAgICAgICAgICAgIHZhcl90ZW1wID0gdmFyKHRlbXApLCAKICAgICAgICAgICAgbWluX3RlbXAgPSBtaW4odGVtcCksIAogICAgICAgICAgICBtYXhfdGVtcCA9IG1heCh0ZW1wKSkgJT4lIAogIG11dGF0ZSgicmFuZ2VfdGVtcCIgPSBtYXhfdGVtcCAtIG1pbl90ZW1wKQoKI3RhYmxlKGhpbmQxX2RhdGEkc3BlY2llcykKCmhpbmQxX2RhdGEgJT4lIAogIGZpbHRlcihsaW1fZGlmZiA8PSA1KSAlPiUgIAogIGdncGxvdChhZXMoeCA9IGRveSwgeSA9IGxpbV9kaWZmLCBjb2xvdXIgPSBzcGVjaWVzKSkgKwogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDApICsgCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gNSwgCiAgICAgICAgICAgICBjb2xvdXIgPSAiZ3JleSIpICsgCiAgZ2VvbV9wb2ludChhbHBoYSA9IDAuNSkgKwogIGdlb21fc21vb3RoKHNlID0gRikgKyAKICBsYWJzKHggPSAiRGF5IG9mIFllYXIiLCAKICAgICAgIHkgPSAiUHJlZGljdGVkIFdhcm1pbmcgVG9sZXJhbmNlIFxuKMKwQyBBYm92ZSBEYWlseSBNYXgpIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQpgYGAKCiMjIyBTY2VuYXJpbyAyCkluIHRoZSBzZWNvbmQgc2NlbmFyaW8sIHRoZXJtYWwgbGltaXRzIHZhcnkgd2l0aGluIGFuZCBiZXR3ZWVuIHNwZWNpZXMuIEEgc2ltcGxlIG1vZGVsIGlzIHVzZWQgdG8gcHJlZGljdCBzcGVjaWVzIHRoZXJtYWwgbGltaXRzIGJhc2VkIG9uIG1lYW4gZGFpbHkgdGVtcGVyYXR1cmUgKENUbWF4IGFzIGEgZnVuY3Rpb24gb2Ygc3BlY2llcyBhbmQgY29sbGVjdGlvbiB0ZW1wZXJhdHVyZSwgYnV0IHdpdGhvdXQgdGhlIGludGVyYWN0aW9uIGJldHdlZW4gdGhlc2UgdHdvIGZhY3RvcnMpLiBUaGVzZSBwcmVkaWN0ZWQgdGhlcm1hbCBsaW1pdHMgYXJlIHRoZW4gY29tcGFyZWQgYWdhaW5zdCB0aGUgbWF4aW11bSBkYWlseSB0ZW1wZXJhdHVyZSB0byBlc3RpbWF0ZSB0aGVybWFsIHN0cmVzcywgYXMgaW4gU2NlbmFyaW8gMS4gSW5jbHVkaW5nIHRoaXMgc2ltcGxlIGZvcm0gb2YgYWNjbGltYXRpb24gaW4gdGhlIG1vZGVsIHJlZHVjZWQgdGhlIGRlZ3JlZSBvZiB0aGVybWFsIHN0cmVzcyBmb3IgZWFjaCBzcGVjaWVzLCBlbGltaW5hdGluZyBpdCBlbnRpcmVseSBmb3IgKkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMqLiBOb3RlIHRoYXQgdGhlIG1hZ25pdHVkZSBvZiB0aGUgcHJlZGljdGVkIHN0cmVzcyBpcyAgbG93IGVub3VnaCB0aGF0IHJlbW92aW5nIHRoZSA1wrBDIGJ1ZmZlciBhcm91bmQgdGhlIHByZWRpY3RlZCB0aGVybWFsIGxpbWl0cyB3b3VsZCBhY3R1YWxseSBsaW1pdCBwcmVkaWN0ZWQgdGhlcm1hbCBzdHJlc3MgdG8ganVzdCBhIGZldyBkYXlzIGZvciAqU2VuZWNlbGxhIGNhbGFub2lkZXMqLiAKCmBgYHtyIGZpZy5oZWlnaHQ9NSwgZmlnLndpZHRoPTEwfQpoaW5kY2FzdF9tb2RlbDEgPSBsbShkYXRhID0gZmlsdGVyKGZ1bGxfZGF0YSwgc2V4ID09ICJmZW1hbGUiKSwKICAgICAgICAgICAgICAgICAgICAgY3RtYXggfiBjb2xsZWN0aW9uX3RlbXAgKyBzcF9uYW1lKQoKaGluZDJfZGF0YSA9IGhpbmRfdGVtcF9kYXRhICU+JSAKICBncm91cF9ieShkYXRlKSAlPiUgCiAgc3VtbWFyaXplKCJjb2xsZWN0aW9uX3RlbXAiID0gbWVhbih0ZW1wKSwKICAgICAgICAgICAgImRhaWx5X21heCIgPSBtYXgodGVtcCkpICU+JSAKICBiaW5kX2NvbHMoCiAgICBwaXZvdF93aWRlcihtZWFuX2N0bWF4LCAKICAgICAgICAgICAgICAgIG5hbWVzX2Zyb20gPSBzcF9uYW1lLCAKICAgICAgICAgICAgICAgIHZhbHVlc19mcm9tID0gbWVhbl9jdG1heCkpICU+JSAKICBwaXZvdF9sb25nZXIoY29scyA9IGMoLWRhdGUsIC1kYWlseV9tYXgsIC1jb2xsZWN0aW9uX3RlbXApLAogICAgICAgICAgICAgICBuYW1lc190byA9ICJzcF9uYW1lIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJtZWFuX2N0bWF4IikgJT4lIAogIHNlbGVjdCgtbWVhbl9jdG1heCkgJT4lIAogIG11dGF0ZSgicHJlZF9jdG1heCIgPSBwcmVkaWN0LmxtIChoaW5kY2FzdF9tb2RlbDEsIG5ld2RhdGEgPSAuKSkgJT4lIAogIHNlbGVjdChkYXRlLCAiZGFpbHlfbWVhbiIgPSBjb2xsZWN0aW9uX3RlbXAsIGRhaWx5X21heCwgInNwZWNpZXMiID0gc3BfbmFtZSwgcHJlZF9jdG1heCkgJT4lIAogIG11dGF0ZShsaW1fZGlmZiA9IHByZWRfY3RtYXggLSBkYWlseV9tYXgpICU+JSAKICAjZmlsdGVyKGxpbV9kaWZmIDw9IDApICU+JSAgCiAgbXV0YXRlKGRveSA9IHlkYXkoZGF0ZSksCiAgICAgICAgICJtZXRob2QiID0gIkNvbnN0YW50X2FjY2xpbWF0aW9uIikKCiMgZ2dwbG90KGhpbmQyX2RhdGEsIGFlcyh4ID0gZGFpbHlfbWVhbiwgeSA9IHByZWRfY3RtYXgsIGNvbG91ciA9IHNwZWNpZXMpKSArCiMgICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iKSAKCiMgdGFibGUoaGluZDJfZGF0YSRzcGVjaWVzKQpoaW5kMl9kYXRhICU+JSAgCiAgZmlsdGVyKGxpbV9kaWZmIDw9IDUpICU+JSAgCiAgZ2dwbG90KGFlcyh4ID0gZG95LCB5ID0gbGltX2RpZmYsIGNvbG91ciA9IHNwZWNpZXMpKSArCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gMCkgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSA1LCAKICAgICAgICAgICAgIGNvbG91ciA9ICJncmV5IikgKyAKICBnZW9tX3BvaW50KGFscGhhID0gMC41KSArCiAgZ2VvbV9zbW9vdGgoKSArIAogIGxhYnMoeCA9ICJEYXkgb2YgWWVhciIsIAogICAgICAgeSA9ICJQcmVkaWN0ZWQgV2FybWluZyBUb2xlcmFuY2UgXG4owrBDIEFib3ZlIERhaWx5IE1heCkiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKIyMjIFNjZW5hcmlvIDMKVGhlIGZpbmFsIHNjZW5hcmlvIGFsbG93cyB0aGUgZW52aXJvbm1lbnRhbCB2YXJpYWJsZSB1c2VkIHRvIHByZWRpY3QgQ1RtYXggdG8gdmFyeSBiZXR3ZWVuIHNwZWNpZXMuIEZvciBzcGVjaWVzIG9ic2VydmVkIGluIGZld2VyIHRoYW4gNSBjb2xsZWN0aW9ucywgd2UgdXNlIHRoZSBzYW1lIGFwcHJvYWNoIGFzIGluIFNjZW5hcmlvIDIuIEZvciBzcGVjaWVzIG9ic2VydmVkIGluIG1vcmUgdGhhbiA1IGNvbGxlY3Rpb25zLCBob3dldmVyLCB0aGUgZmFjdG9yIHdpdGggdGhlIHN0cm9uZ2VzdCBjb3JyZWxhdGlvbiB3aXRoIENUbWF4IGlzIHVzZWQgdG8gcHJlZGljdCB0aGVybWFsIGxpbWl0cy4gVGhlc2UgZmFjdG9ycyBhcmUgaW5jbHVkZWQgYmVsb3cuCgpgYGB7cn0KaGluZF9wcmVkcyA9IGNvcnJfdmFscyAlPiUgIAogIGZpbHRlcihzaWcgPT0gIlNpZy4iKSAlPiUgCiAgZHJvcF9uYShjb3JyZWxhdGlvbikgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JQogIGFycmFuZ2UoZGVzYyhjb3JyZWxhdGlvbikpICU+JSAKICBzbGljZV9oZWFkKG4gPSAxKSAlPiUgCiAgc2VsZWN0KCJTcGVjaWVzIiA9IHNwX25hbWUsICJQcmVkaWN0b3IiID0gcGFyYW1ldGVyLCAiRHVyYXRpb24iID0gZHVyYXRpb24sICJDb3JyZWxhdGlvbiIgPSBjb3JyZWxhdGlvbiwgIlAtVmFsdWUiID0gcC52YWx1ZSkKCmtuaXRyOjprYWJsZShoaW5kX3ByZWRzLCBhbGlnbiA9ICJjIikKYGBgCgpgYGB7cn0KaGluZDNfZGF0YSA9IGhpbmQyX2RhdGEgJT4lICMgQ29udGFpbnMgZGF0YSBmb3Igc3BlY2llcyB0aGF0IHdvbid0IGNoYW5nZSBmcm9tIHNjZW5hcmlvIDIKICBmaWx0ZXIoIShzcGVjaWVzICVpbiUgY29ycl92YWxzJHNwX25hbWUpKQoKcHJlZHNfdG9fcHVsbCA9IGhpbmRfcHJlZHMgJT4lICAKICBzZWxlY3QoU3BlY2llcywgUHJlZGljdG9yLCBEdXJhdGlvbikgCgpmb3IoaSBpbiAxOmxlbmd0aChwcmVkc190b19wdWxsJFNwZWNpZXMpKXsKICAKICBpZihwcmVkc190b19wdWxsJER1cmF0aW9uW2ldID09ICJwcmlvciIpeyAjVGhlIHByaW9yIGRheSB0ZW1wZXJhdHVyZSBtZXRyaWNzIHNob3VsZCBiZSB1c2VkCiAgICBkdXJhdGlvbiA9IE5BCiAgICAKICAgIHByZWRpY3RvcnMgPSBoaW5kX2RhaWx5X3RlbXBfZGF0YSAlPiUgCiAgICAgIG11dGF0ZShkYXRlID0gZGF0ZSArIDEpIAogICAgCiAgICBwYXJhbWV0ZXIgPSBwcmVkc190b19wdWxsJFByZWRpY3RvcltpXQogICAgCiAgICBtb2RlbF9kYXRhID0gZnVsbF9kYXRhICU+JQogICAgICBmaWx0ZXIoc3BfbmFtZSAlaW4lIHByZWRzX3RvX3B1bGwkU3BlY2llc1tpXSkgJT4lIAogICAgICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgICAgIG11dGF0ZShjb2xsZWN0aW9uX2RhdGUgPSBhc19kYXRlKGNvbGxlY3Rpb25fZGF0ZSkpICU+JSAKICAgICAgaW5uZXJfam9pbihwcmVkaWN0b3JzLCBqb2luX2J5KGNvbGxlY3Rpb25fZGF0ZSA9PSBkYXRlKSkgJT4lICAKICAgICAgc2VsZWN0KGN0bWF4LCBjb250YWlucyhwYXJhbWV0ZXIpKQogICAgCiAgICBpZihkaW0obW9kZWxfZGF0YSlbMl0gPT0gMil7CiAgICAgIGhpbmQubW9kZWwgPSBsbShkYXRhID0gbW9kZWxfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICBjdG1heCB+IC4pCiAgICAgIAogICAgICBzcF9kYXRhID0gcHJlZGljdG9ycyAlPiUgCiAgICAgICAgc2VsZWN0KGRhdGUsIGNvbnRhaW5zKHBhcmFtZXRlcikpICU+JSAKICAgICAgICBtdXRhdGUocHJlZF9jdG1heCA9IHByZWRpY3QoaGluZC5tb2RlbCwgbmV3ZGF0YSA9IC4pKSAlPiUgIAogICAgICAgIHNlbGVjdChkYXRlLCBwcmVkX2N0bWF4KSAlPiUgCiAgICAgICAgaW5uZXJfam9pbihoaW5kX2RhaWx5X3RlbXBfZGF0YSwgYnkgPSBjKCJkYXRlIikpICU+JSAKICAgICAgICBtdXRhdGUoInNwZWNpZXMiID0gcHJlZHNfdG9fcHVsbCRTcGVjaWVzW2ldLAogICAgICAgICAgICAgICAiZG95IiA9IHlkYXkoZGF0ZSksCiAgICAgICAgICAgICAgIGxpbV9kaWZmID0gcHJlZF9jdG1heCAtIG1heF90ZW1wKSAlPiUgCiAgICAgICAgc2VsZWN0KGRhdGUsIGRhaWx5X21lYW4gPSBtZWFuX3RlbXAsIGRhaWx5X21heCA9IG1heF90ZW1wLCBzcGVjaWVzLCBwcmVkX2N0bWF4LCBsaW1fZGlmZiwgZG95KQogICAgICAKICAgICAgaGluZDNfZGF0YSA9IGJpbmRfcm93cyhoaW5kM19kYXRhLCBzcF9kYXRhKQogICAgfWVsc2V7CiAgICAgIHByaW50KCJUb28gbWFueSBjb2x1bW5zIHNlbGVjdGVkIikKICAgIH0KICAgIAogICAgCiAgfWVsc2V7CiAgICBkdXJhdGlvbiA9IGFzLm51bWVyaWMocHJlZHNfdG9fcHVsbCREdXJhdGlvbltpXSkKICB9CiAgCiAgaWYocHJlZHNfdG9fcHVsbCREdXJhdGlvbltpXSAhPSAicHJpb3IiICYgaXMubmEoZHVyYXRpb24pKXsgI0RhaWx5IHRlbXBlcmF0dXJlcyBzaG91bGQgYmUgdXNlZCwgYXMgaW4gU2NlbmFyaW8gMgogICAgc3BfZGF0YSA9IGhpbmQyX2RhdGEgJT4lIAogICAgICBmaWx0ZXIoc3BlY2llcyA9PSBwcmVkc190b19wdWxsJFNwZWNpZXNbaV0pCiAgICAKICAgIGhpbmQzX2RhdGEgPSBiaW5kX3Jvd3MoaGluZDNfZGF0YSwgc3BfZGF0YSkKICB9CiAgCiAgaWYoaXMubnVtZXJpYyhkdXJhdGlvbikpewogICAgI05laXRoZXIgdGhlIHByaW9yIGRheSBub3IgZGF5IG9mIG1ldHJpY3Mgc2hvdWxkIGJlIHVzZWQ7IHVzZSBkdXJhdGlvbiBhcyBuX2RheXMKICAgIAogICAgcHJlZGljdG9ycyA9IGdldF9wcmVkaWN0b3JzKGRhaWx5X3ZhbHVlcyA9IGhpbmRfZGFpbHlfdGVtcF9kYXRhLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfdGVtcCA9IGhpbmRfdGVtcF9kYXRhLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuX2RheXMgPSBkdXJhdGlvbikKICAgIAogICAgcGFyYW1ldGVyID0gcHJlZHNfdG9fcHVsbCRQcmVkaWN0b3JbaV0KICAgIAogICAgbW9kZWxfZGF0YSA9IGZ1bGxfZGF0YSAlPiUKICAgICAgZmlsdGVyKHNwX25hbWUgJWluJSBwcmVkc190b19wdWxsJFNwZWNpZXNbaV0pICU+JSAKICAgICAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogICAgICBtdXRhdGUoY29sbGVjdGlvbl9kYXRlID0gYXNfZGF0ZShjb2xsZWN0aW9uX2RhdGUpKSAlPiUgCiAgICAgIGlubmVyX2pvaW4ocHJlZGljdG9ycywgam9pbl9ieShjb2xsZWN0aW9uX2RhdGUgPT0gZGF0ZSkpICU+JSAgCiAgICAgIHNlbGVjdChjdG1heCwgY29udGFpbnMocGFzdGUoImRheV8iLCBwYXJhbWV0ZXIsIHNlcCA9ICIiKSkpCiAgICAKICAgIGlmKGRpbShtb2RlbF9kYXRhKVsyXSA9PSAyKXsKICAgICAgaGluZC5tb2RlbCA9IGxtKGRhdGEgPSBtb2RlbF9kYXRhLCAKICAgICAgICAgICAgICAgICAgICAgIGN0bWF4IH4gLikKICAgICAgCiAgICAgIHNwX2RhdGEgPSBwcmVkaWN0b3JzICU+JSAKICAgICAgICBzZWxlY3QoZGF0ZSwgY29udGFpbnMocGFyYW1ldGVyKSkgJT4lIAogICAgICAgIG11dGF0ZShwcmVkX2N0bWF4ID0gcHJlZGljdChoaW5kLm1vZGVsLCBuZXdkYXRhID0gLikpICU+JSAgCiAgICAgICAgc2VsZWN0KGRhdGUsIHByZWRfY3RtYXgpICU+JSAKICAgICAgICBpbm5lcl9qb2luKGhpbmRfZGFpbHlfdGVtcF9kYXRhLCBieSA9IGMoImRhdGUiKSkgJT4lIAogICAgICAgIG11dGF0ZSgic3BlY2llcyIgPSBwcmVkc190b19wdWxsJFNwZWNpZXNbaV0sCiAgICAgICAgICAgICAgICJkb3kiID0geWRheShkYXRlKSwKICAgICAgICAgICAgICAgbGltX2RpZmYgPSBwcmVkX2N0bWF4IC0gbWF4X3RlbXApICU+JSAKICAgICAgICBzZWxlY3QoZGF0ZSwgZGFpbHlfbWVhbiA9IG1lYW5fdGVtcCwgZGFpbHlfbWF4ID0gbWF4X3RlbXAsIHNwZWNpZXMsIHByZWRfY3RtYXgsIGxpbV9kaWZmLCBkb3kpCiAgICAgIAogICAgICBoaW5kM19kYXRhID0gYmluZF9yb3dzKGhpbmQzX2RhdGEsIHNwX2RhdGEpCiAgICAgIAogICAgfWVsc2V7CiAgICAgIHByaW50KCJUb28gbWFueSBjb2x1bW5zIHNlbGVjdGVkIikKICAgIH0KICAgIAogIH0KfQoKaGluZDNfZGF0YSA9IGhpbmQzX2RhdGEgJT4lIAogIG11dGF0ZSgibWV0aG9kIiA9ICJWYXJpYWJsZV9hY2NsaW1hdGlvbiIpCmBgYAoKVGhpcyB0aGlyZCBhcHByb2FjaCBkaWQgbm90IGFmZmVjdCB0aGUgcHJlZGljdGVkIHBhdHRlcm5zIGluICpMaW1ub2NhbGFudXMqIG9yICpTZW5lY2VsbGEqIChuZWl0aGVyIHNwZWNpZXMgaGFzIGJlZW4gb2JzZXJ2ZWQgaW4gZW5vdWdoIGNvbGxlY3Rpb25zIHRvIGVzdGltYXRlIHRoZSBlZmZlY3RzIG9mIGRpZmZlcmVudCBlbnZpcm9ubWVudGFsIGZhY3RvcnMpLiBDaGFuZ2luZyB0aGUgYWNjbGltYXRpb24gYXBwcm9hY2ggZGlkIGFmZmVjdCBwYXR0ZXJucyBpbiB0aGVybWFsIGxpbWl0cyBpbiB0aGUgb3RoZXIgc3BlY2llcyB0aG91Z2guIFRoZSBmaWd1cmUgYmVsb3cgc2hvd3MgaG93IHByZWRpY3RlZCB3YXJtaW5nIHRvbGVyYW5jZSB2YXJpZXMgb3ZlciB0aGUgeWVhciBpbiB0aGUgc2V2ZW4gc3BlY2llcywgYmFzZWQgb24gdGhlIHRocmVlIGRpZmZlcmVudCBwcmVkaWN0aW9uIG1ldGhvZHMuIEluIGdlbmVyYWwsIGNvbnN0YW50IHRoZXJtYWwgbGltaXRzICh0aGUgJ25vIGFjY2xpbWF0aW9uJyBtZXRob2QpIHJlc3VsdGVkIGluIGxhcmdlciB3YXJtaW5nIHRvbGVyYW5jZSBkdXJpbmcgdGhlIHdpbnRlciBhbmQgbG93ZXIgd2FybWluZyB0b2xlcmFuY2UgZHVyaW5nIHRoZSBzdW1tZXIsIGFsdGhvdWdoIHRoaXMgZWZmZWN0IHdhcyBzbWFsbCBpbiBtb3N0IHNwZWNpZXMuICAgICAKCmBgYHtyIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD0xMH0Kc3ludGhlc2lzID0gYmluZF9yb3dzKAogIHNlbGVjdChoaW5kMV9kYXRhLCBkYXRlLCBkb3ksIGRhaWx5X21lYW4sIGRhaWx5X21heCwgc3BlY2llcywgInByZWRfY3RtYXgiID0gbWVhbl9jdG1heCwgbGltX2RpZmYsIG1ldGhvZCksCiAgc2VsZWN0KGhpbmQyX2RhdGEsIGRhdGUsIGRveSwgZGFpbHlfbWVhbiwgZGFpbHlfbWF4LCAgc3BlY2llcywgcHJlZF9jdG1heCwgbGltX2RpZmYsIG1ldGhvZCksCiAgc2VsZWN0KGhpbmQzX2RhdGEsIGRhdGUsIGRveSwgZGFpbHlfbWVhbiwgZGFpbHlfbWF4LCAgc3BlY2llcywgcHJlZF9jdG1heCwgbGltX2RpZmYsIG1ldGhvZCkpICU+JSAKICBtdXRhdGUobWV0aG9kID0gZmN0X3JlbGV2ZWwobWV0aG9kLCAiTm9fYWNjbGltYXRpb24iLCAiQ29uc3RhbnRfYWNjbGltYXRpb24iLCAiVmFyaWFibGVfYWNjbGltYXRpb24iKSkKCmNsaW1hdG9sb2d5ID0gc3ludGhlc2lzICU+JSAKICBncm91cF9ieShzcGVjaWVzLCBkb3ksIG1ldGhvZCkgJT4lICAKICBzdW1tYXJpc2UoIm1lYW5fZGlmZiIgPSBtZWFuKGxpbV9kaWZmKSwKICAgICAgICAgICAgIm1pbl9kaWZmIiA9IG1pbihsaW1fZGlmZiksCiAgICAgICAgICAgICJtYXhfZGlmZiIgPSBtYXgobGltX2RpZmYpKSAlPiUgCiAgbXV0YXRlKG1ldGhvZCA9IGZjdF9yZWxldmVsKG1ldGhvZCwgIk5vX2FjY2xpbWF0aW9uIiwgIkNvbnN0YW50X2FjY2xpbWF0aW9uIiwgIlZhcmlhYmxlX2FjY2xpbWF0aW9uIikpCgphY2NfZWZmZWN0cyA9IHN5bnRoZXNpcyAlPiUgCiAgcGl2b3Rfd2lkZXIoaWRfY29scyA9IGMoZGF0ZSwgc3BlY2llcywgZG95KSwgCiAgICAgICAgICAgICAgbmFtZXNfZnJvbSA9IG1ldGhvZCwgCiAgICAgICAgICAgICAgdmFsdWVzX2Zyb20gPSBsaW1fZGlmZikgJT4lICAKICBtdXRhdGUoImNvbnN0X2FjY19lZmZlY3QiID0gQ29uc3RhbnRfYWNjbGltYXRpb24gLSBOb19hY2NsaW1hdGlvbiwKICAgICAgICAgInZhcl9hY2NfZWZmZWN0IiA9IFZhcmlhYmxlX2FjY2xpbWF0aW9uIC0gTm9fYWNjbGltYXRpb24pCgpnZ3Bsb3Qoc3ludGhlc2lzLCBhZXMoeCA9IGRveSwgeSA9IGxpbV9kaWZmLCBjb2xvdXIgPSBtZXRob2QpKSArIAogIGZhY2V0X3dyYXAoc3BlY2llc34uKSArIAogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDApICsgCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gNSwgY29sb3VyID0gImdyZXkiKSArIAogIGdlb21fcG9pbnQoYWxwaGEgPSAwLjEpICsgCiAgbGFicyh4ID0gIkRheSBvZiBZZWFyIiwgCiAgICAgICB5ID0gIlByZWRpY3RlZCBXYXJtaW5nIFRvbGVyYW5jZSAowrBDIEFib3ZlIERhaWx5IE1heCkiKSArIAogIGd1aWRlcyhjb2xvdXIgPSBndWlkZV9sZWdlbmQob3ZlcnJpZGUuYWVzID0gbGlzdChhbHBoYSA9IDEpKSkgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cyhiYXNlX3NpemUgPSAxOCkgKyAKICB0aGVtZShzdHJpcC50ZXh0LngudG9wID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMCkpCmBgYAoKYGBge3IsIGluY2x1ZGUgPSBGfQp5ZWFybHlfc3VtbWFyeSA9IHN5bnRoZXNpcyAlPiUgIAogIG11dGF0ZSgieWVhciIgPSB5ZWFyKGRhdGUpKSAlPiUgCiAgZ3JvdXBfYnkoc3BlY2llcywgeWVhciwgbWV0aG9kKSAlPiUgCiAgc3VtbWFyaXNlKCJtaW5fd3QiID0gbWluKGxpbV9kaWZmKSwKICAgICAgICAgICAgIm1heF93dCIgPSBtYXgobGltX2RpZmYpKSAlPiUgCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKG1pbl93dCwgbWF4X3d0KSwgCiAgICAgICAgICAgICAgIG5hbWVzX3RvID0gIm1ldHJpYyIsIAogICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAid3QiKQoKZ2dwbG90KHllYXJseV9zdW1tYXJ5LCBhZXMoeCA9IG1ldGhvZCwgeSA9IHd0LCBjb2xvdXIgPSBtZXRyaWMpKSArIAogIGZhY2V0X3dyYXAoLn5zcGVjaWVzKSArIAogIGdlb21fcG9pbnQoKSArIAogIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDMwMCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSkpCmBgYAoK